diff --git a/src/game/BattleGround/BattleGround.cpp b/src/game/BattleGround/BattleGround.cpp index 8583f849bb..dd3374958a 100644 --- a/src/game/BattleGround/BattleGround.cpp +++ b/src/game/BattleGround/BattleGround.cpp @@ -263,7 +263,12 @@ BattleGround::~BattleGround() // skip template bgs as they were never added to visible bg list BattleGroundBracketId bracketId = GetBracketId(); if (bracketId != BG_BRACKET_ID_TEMPLATE) - sBattleGroundMgr.DeleteClientVisibleInstanceId(GetTypeId(), bracketId, GetClientInstanceId()); + { + sWorld.GetBGQueue().GetMessager().AddMessage([bgTypeId = GetTypeId(), bracketId, clientInstanceId = GetClientInstanceId()](BattleGroundQueue* queue) + { + queue->DeleteClientVisibleInstanceId(bgTypeId, bracketId, clientInstanceId); + }); + } // unload map // map can be null at bg destruction @@ -271,7 +276,7 @@ BattleGround::~BattleGround() m_bgMap->SetUnload(); // remove from bg free slot queue - this->RemoveFromBgFreeSlotQueue(); + this->RemovedFromBgFreeSlotQueue(true); for (BattleGroundScoreMap::const_iterator itr = m_playerScores.begin(); itr != m_playerScores.end(); ++itr) delete itr->second; @@ -691,7 +696,7 @@ void BattleGround::UpdateWorldStateForPlayer(uint32 field, uint32 value, Player* */ void BattleGround::EndBattleGround(Team winner) { - this->RemoveFromBgFreeSlotQueue(); + this->RemovedFromBgFreeSlotQueue(true); uint32 loser_rating = 0; uint32 winner_rating = 0; @@ -832,7 +837,7 @@ void BattleGround::EndBattleGround(Team winner) plr->GetSession()->SendPacket(data); BattleGroundQueueTypeId bgQueueTypeId = BattleGroundMgr::BgQueueTypeId(GetTypeId()); - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, this, plr->GetBattleGroundQueueIndex(bgQueueTypeId), STATUS_IN_PROGRESS, TIME_TO_AUTOREMOVE, GetStartTime()); + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, GetTypeId(), GetClientInstanceId(), GetMapId(), plr->GetBattleGroundQueueIndex(bgQueueTypeId), STATUS_IN_PROGRESS, TIME_TO_AUTOREMOVE, GetStartTime()); plr->GetSession()->SendPacket(data); } @@ -852,6 +857,16 @@ uint32 BattleGround::GetBonusHonorFromKill(uint32 kills) const return (uint32)MaNGOS::Honor::hk_honor_at_level(GetMaxLevel(), kills); } +void BattleGround::SetStatus(BattleGroundStatus status) +{ + m_status = status; + sWorld.GetBGQueue().GetMessager().AddMessage([status, bgTypeId = GetTypeId(), instanceId = GetInstanceId()](BattleGroundQueue* queue) + { + if (BattleGroundInQueueInfo* bgInstance = queue->GetFreeSlotInstance(bgTypeId, instanceId)) + bgInstance->status = status; + }); +} + /** Function that returns the battleground master entry */ @@ -1084,7 +1099,7 @@ void BattleGround::RemovePlayerAtLeave(ObjectGuid playerGuid, bool isOnTransport if (doSendPacket) { WorldPacket data; - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, this, player->GetBattleGroundQueueIndex(bgQueueTypeId), STATUS_NONE, 0, 0); + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, GetTypeId(), GetClientInstanceId(), GetMapId(), player->GetBattleGroundQueueIndex(bgQueueTypeId), STATUS_NONE, 0, 0); player->GetSession()->SendPacket(data); } @@ -1101,13 +1116,24 @@ void BattleGround::RemovePlayerAtLeave(ObjectGuid playerGuid, bool isOnTransport delete group; } } - DecreaseInvitedCount(team); + + SetInvitedCount(team, GetInvitedCount(team) - 1); // change ahead of free slot queue - will be synched again after // we should update battleground queue, but only if bg isn't ending if (GetStatus() < STATUS_WAIT_LEAVE) { // a player has left the battleground, so there are free slots -> add to queue - AddToBgFreeSlotQueue(); - sBattleGroundMgr.ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, GetBracketId()); + if (!AddToBgFreeSlotQueue()) // avoid setting two messages - if was already in queue, just update count + { + sWorld.GetBGQueue().GetMessager().AddMessage([bgTypeId, instanceId = GetInstanceId(), team](BattleGroundQueue* queue) + { + if (BattleGroundInQueueInfo* bgInstance = queue->GetFreeSlotInstance(bgTypeId, instanceId)) + bgInstance->DecreaseInvitedCount(team); + }); + } + sWorld.GetBGQueue().GetMessager().AddMessage([bgQueueTypeId, bgTypeId, bracketId = GetBracketId()](BattleGroundQueue* queue) + { + queue->ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, bracketId); + }); } // Let others know @@ -1168,8 +1194,7 @@ void BattleGround::StartBattleGround() { SetStartTime(0); - // add BG to free slot queue - AddToBgFreeSlotQueue(); + // expects to be already added in free queue // add bg to update list // This must be done here, because we need to have already invited some players when first BG::Update() method is executed @@ -1293,33 +1318,45 @@ void BattleGround::EventPlayerLoggedOut(Player* player) /** Function that returns the number of players that can join a battleground based on the provided team */ -void BattleGround::AddToBgFreeSlotQueue() +bool BattleGround::AddToBgFreeSlotQueue() { // make sure to add only once if (!m_hasBgFreeSlotQueue) { - sBattleGroundMgr.BgFreeSlotQueue[m_typeId].push_front(this); m_hasBgFreeSlotQueue = true; + BattleGroundInQueueInfo bgInfo; + bgInfo.Fill(this); + sWorld.GetBGQueue().GetMessager().AddMessage([bgInfo](BattleGroundQueue* queue) + { + queue->AddBgToFreeSlots(bgInfo); + }); + return true; } + return false; } /** Method that removes this battleground from free queue - it must be called when deleting battleground */ -void BattleGround::RemoveFromBgFreeSlotQueue() +void BattleGround::RemovedFromBgFreeSlotQueue(bool removeFromQueue) { // set to be able to re-add if needed - m_hasBgFreeSlotQueue = false; - BgFreeSlotQueueType& bgFreeSlot = sBattleGroundMgr.BgFreeSlotQueue[m_typeId]; - - for (BgFreeSlotQueueType::iterator itr = bgFreeSlot.begin(); itr != bgFreeSlot.end(); ++itr) + if (m_hasBgFreeSlotQueue && removeFromQueue) { - if ((*itr)->GetInstanceId() == GetInstanceId()) + sWorld.GetBGQueue().GetMessager().AddMessage([bgTypeId = GetTypeId(), instanceId = GetInstanceId()](BattleGroundQueue* queue) { - bgFreeSlot.erase(itr); - return; - } + queue->RemoveBgFromFreeSlots(bgTypeId, instanceId); + }); } + m_hasBgFreeSlotQueue = false; +} + +void BattleGround::SetInvitedCount(Team team, uint32 count) +{ + if (team == ALLIANCE) + m_invitedAlliance = count; + else + m_invitedHorde = count; } /** @@ -1440,7 +1477,7 @@ Team BattleGround::GetPrematureWinner() */ void BattleGround::OnObjectDBLoad(Creature* creature) { - const BattleGroundEventIdx eventId = sBattleGroundMgr.GetCreatureEventIndex(creature->GetDbGuid()); + const BattleGroundEventIdx eventId = GetBgMap()->GetMapDataContainer().GetCreatureEventIndex(creature->GetDbGuid()); if (eventId.event1 == BG_EVENT_NONE) return; @@ -1488,7 +1525,7 @@ uint32 BattleGround::GetSingleGameObjectGuid(uint8 event1, uint8 event2) */ void BattleGround::OnObjectDBLoad(GameObject* obj) { - const BattleGroundEventIdx eventId = sBattleGroundMgr.GetGameObjectEventIndex(obj->GetDbGuid()); + const BattleGroundEventIdx eventId = GetBgMap()->GetMapDataContainer().GetGameObjectEventIndex(obj->GetDbGuid()); if (eventId.event1 == BG_EVENT_NONE) return; @@ -1745,7 +1782,7 @@ void BattleGround::SendBcdToTeam(int32 bcdEntry, ChatMsg msgtype, Creature const */ void BattleGround::EndNow() { - RemoveFromBgFreeSlotQueue(); + RemovedFromBgFreeSlotQueue(true); SetStatus(STATUS_WAIT_LEAVE); SetEndTime(0); } @@ -1838,6 +1875,8 @@ void BattleGround::PlayerAddedToBgCheckIfBgIsRunning(Player* player) sBattleGroundMgr.BuildPvpLogDataPacket(data, this); player->GetSession()->SendPacket(data); + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, GetTypeId(), GetClientInstanceId(), GetMapId(), player->GetBattleGroundQueueIndex(bgQueueTypeId), STATUS_IN_PROGRESS, GetEndTime(), GetStartTime()); + player->GetSession()->SendPacket(data); } /** diff --git a/src/game/BattleGround/BattleGround.h b/src/game/BattleGround/BattleGround.h index 3fffcfcae1..dabefd4a97 100644 --- a/src/game/BattleGround/BattleGround.h +++ b/src/game/BattleGround/BattleGround.h @@ -24,6 +24,7 @@ #include "Maps/Map.h" #include "Util/ByteBuffer.h" #include "Entities/ObjectGuid.h" +#include "BattleGround/BattleGroundDefines.h" // magic event-numbers #define BG_EVENT_NONE 255 @@ -298,9 +299,9 @@ class BattleGround // Set methods: void SetName(char const* name) { m_name = name; } - void SetTypeID(BattleGroundTypeId typeId) { m_typeId = typeId; } - void SetBracketId(BattleGroundBracketId Id) { m_bracketId = Id; } - void SetStatus(BattleGroundStatus status) { m_status = status; } + void SetTypeId(BattleGroundTypeId typeId) { m_typeId = typeId; } + void SetBracketId(BattleGroundBracketId id) { m_bracketId = id; } + void SetStatus(BattleGroundStatus status); void SetClientInstanceId(uint32 instanceId) { m_clientInstanceId = instanceId; } void SetStartTime(uint32 time) { m_startTime = time; } void SetEndTime(uint32 time) { m_endTime = time; } @@ -315,11 +316,11 @@ class BattleGround void SetMaxPlayersPerTeam(uint32 maxPlayers) { m_maxPlayersPerTeam = maxPlayers; } void SetMinPlayersPerTeam(uint32 minPlayers) { m_minPlayersPerTeam = minPlayers; } - void AddToBgFreeSlotQueue(); // this queue will be useful when more battlegrounds instances will be available - void RemoveFromBgFreeSlotQueue(); // this method could delete whole BG instance, if another free is available + bool AddToBgFreeSlotQueue(); // this queue will be useful when more battlegrounds instances will be available + void RemovedFromBgFreeSlotQueue(bool removeFromQueue); // this method could delete whole BG instance, if another free is available - void DecreaseInvitedCount(Team team) { (team == ALLIANCE) ? --m_invitedAlliance : --m_invitedHorde; } - void IncreaseInvitedCount(Team team) { (team == ALLIANCE) ? ++m_invitedAlliance : ++m_invitedHorde; } + // Functions to decrease or increase player count + void SetInvitedCount(Team team, uint32 count); uint32 GetInvitedCount(Team team) const { if (team == ALLIANCE) @@ -540,7 +541,7 @@ class BattleGround // returns the other team index static PvpTeamIndex GetOtherTeamIndex(PvpTeamIndex teamIdx) { return teamIdx == TEAM_INDEX_ALLIANCE ? TEAM_INDEX_HORDE : TEAM_INDEX_ALLIANCE; } - // checke if player is inside battleground + // check if player is inside battleground bool IsPlayerInBattleGround(ObjectGuid /*playerGuid*/); // Handle script condition fulfillment diff --git a/src/game/BattleGround/BattleGroundAB.cpp b/src/game/BattleGround/BattleGroundAB.cpp index 8088ed603d..0233c64d66 100644 --- a/src/game/BattleGround/BattleGroundAB.cpp +++ b/src/game/BattleGround/BattleGroundAB.cpp @@ -268,7 +268,7 @@ void BattleGroundAB::HandlePlayerClickedOnFlag(Player* player, GameObject* go) uint32 factionStrig = 0; // process battleground event - uint8 event = (sBattleGroundMgr.GetGameObjectEventIndex(go->GetDbGuid())).event1; + uint8 event = (GetBgMap()->GetMapDataContainer().GetGameObjectEventIndex(go->GetDbGuid())).event1; if (event >= BG_AB_MAX_NODES) // not a node return; diff --git a/src/game/BattleGround/BattleGroundAV.cpp b/src/game/BattleGround/BattleGroundAV.cpp index a49c3bbc8a..fd98eaa603 100644 --- a/src/game/BattleGround/BattleGroundAV.cpp +++ b/src/game/BattleGround/BattleGroundAV.cpp @@ -616,12 +616,12 @@ void BattleGroundAV::HandlePlayerClickedOnFlag(Player* player, GameObject* go) DEBUG_LOG("BattleGroundAV: Player from team %u clicked on gameobject entry %u", player->GetTeam(), go->GetEntry()); - uint8 event = (sBattleGroundMgr.GetGameObjectEventIndex(go->GetDbGuid())).event1; + uint8 event = (GetBgMap()->GetMapDataContainer().GetGameObjectEventIndex(go->GetDbGuid())).event1; if (event >= BG_AV_MAX_NODES) // not a node return; AVNodeIds node = AVNodeIds(event); - switch ((sBattleGroundMgr.GetGameObjectEventIndex(go->GetDbGuid())).event2 % BG_AV_MAX_STATES) + switch ((GetBgMap()->GetMapDataContainer().GetGameObjectEventIndex(go->GetDbGuid())).event2 % BG_AV_MAX_STATES) { case POINT_CONTROLLED: ProcessPlayerAssaultsPoint(player, node); diff --git a/src/game/BattleGround/BattleGroundDefines.h b/src/game/BattleGround/BattleGroundDefines.h new file mode 100644 index 0000000000..a5a7c8c90e --- /dev/null +++ b/src/game/BattleGround/BattleGroundDefines.h @@ -0,0 +1,94 @@ +/* + * This file is part of the CMaNGOS Project. See AUTHORS file for Copyright information + * + * 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 + * (at your option) 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, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#ifndef _BG_DEFINES_H +#define _BG_DEFINES_H + +#include "Common.h" + +#define BATTLEGROUND_ARENA_POINT_DISTRIBUTION_DAY 86400 // seconds in a day +#define COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME 10 + +enum BattleGroundQueueGroupTypes +{ + BG_QUEUE_PREMADE_ALLIANCE = 0, + BG_QUEUE_PREMADE_HORDE = 1, + BG_QUEUE_NORMAL_ALLIANCE = 2, + BG_QUEUE_NORMAL_HORDE = 3 +}; + +#define BG_QUEUE_GROUP_TYPES_COUNT 4 + +enum BattleGroundGroupJoinStatus +{ + BG_GROUP_JOIN_STATUS_TEAM_LEFT_QUEUE = -7, + BG_GROUP_JOIN_STATUS_QUEUED_FOR_RATED = -6, + BG_GROUP_JOIN_STATUS_CANNOT_QUEUE_FOR_RATED = -5, + BG_GROUP_JOIN_STATUS_TOO_MANY_QUEUES = -4, + BG_GROUP_JOIN_STATUS_NOT_IN_TEAM = -3, + BG_GROUP_JOIN_STATUS_DESERTERS = -2, + BG_GROUP_JOIN_STATUS_NOT_ELIGIBLE = -1, + BG_GROUP_JOIN_STATUS_SUCCESS = 0, +}; + +// indexes of BattlemasterList.dbc +enum BattleGroundTypeId +{ + BATTLEGROUND_TYPE_NONE = 0, + BATTLEGROUND_AV = 1, + BATTLEGROUND_WS = 2, + BATTLEGROUND_AB = 3, +}; +#define MAX_BATTLEGROUND_TYPE_ID 4 + +inline BattleGroundTypeId GetBattleGroundTypeIdByMapId(uint32 mapId) +{ + switch (mapId) + { + case 30: return BATTLEGROUND_AV; + case 489: return BATTLEGROUND_WS; + case 529: return BATTLEGROUND_AB; + default: return BATTLEGROUND_TYPE_NONE; + } +} + +inline uint32 GetBattleGrounMapIdByTypeId(BattleGroundTypeId bgTypeId) +{ + switch (bgTypeId) + { + case BATTLEGROUND_AV: return 30; + case BATTLEGROUND_WS: return 489; + case BATTLEGROUND_AB: return 529; + default: return 0; // none + } + + // impossible, just make compiler happy + return 0; +} + +enum ArenaType +{ + ARENA_TYPE_NONE = 0, // used for mark non-arenas or problematic cases + ARENA_TYPE_2v2 = 2, + ARENA_TYPE_3v3 = 3, + ARENA_TYPE_5v5 = 5 +}; + +inline bool IsArenaTypeValid(ArenaType type) { return type == ARENA_TYPE_2v2 || type == ARENA_TYPE_3v3 || type == ARENA_TYPE_5v5; } + +#endif \ No newline at end of file diff --git a/src/game/BattleGround/BattleGroundHandler.cpp b/src/game/BattleGround/BattleGroundHandler.cpp index 4fc2abd6ef..cf4df89d4b 100644 --- a/src/game/BattleGround/BattleGroundHandler.cpp +++ b/src/game/BattleGround/BattleGroundHandler.cpp @@ -52,7 +52,7 @@ void WorldSession::HandleBattlemasterHelloOpcode(WorldPacket& recv_data) if (uint32 pauseTimer = pCreature->GetInteractionPauseTimer()) pCreature->GetMotionMaster()->PauseWaypoints(pauseTimer); - BattleGroundTypeId bgTypeId = sBattleGroundMgr.GetBattleMasterBG(pCreature->GetEntry()); + BattleGroundTypeId bgTypeId = GetPlayer()->GetMap()->GetMapDataContainer().GetBattleMasterBG(pCreature->GetEntry()); if (bgTypeId == BATTLEGROUND_TYPE_NONE) return; @@ -75,9 +75,16 @@ Send battleground list */ void WorldSession::SendBattleGroundList(ObjectGuid guid, BattleGroundTypeId bgTypeId) const { - WorldPacket data; - sBattleGroundMgr.BuildBattleGroundListPacket(data, guid, _player, bgTypeId); - SendPacket(data); + sWorld.GetBGQueue().GetMessager().AddMessage([playerGuid = _player->GetObjectGuid(), masterGuid = guid, playerLevel = _player->GetLevel(), bgTypeId](BattleGroundQueue* queue) + { + WorldPacket data; + queue->BuildBattleGroundListPacket(data, masterGuid, playerLevel, BattleGroundTypeId(bgTypeId)); + sWorld.GetMessager().AddMessage([playerGuid, data](World* world) + { + if (Player* player = sObjectMgr.GetPlayer(playerGuid)) + player->GetSession()->SendPacket(data); + }); + }); } // Sent by client when player wants to join a battleground @@ -88,7 +95,7 @@ void WorldSession::HandleBattlemasterJoinOpcode(WorldPacket& recv_data) uint32 mapId; uint8 joinAsGroup; bool isPremade = false; - Group* grp = nullptr; + Group* group = nullptr; recv_data >> guid; // battlemaster guid recv_data >> mapId; @@ -133,7 +140,9 @@ void WorldSession::HandleBattlemasterJoinOpcode(WorldPacket& recv_data) return; } - BattleGroundBracketId bgBracketId = _player->GetBattleGroundBracketIdFromLevel(bgTypeId); + mapId = bg->GetMapId(); + + BattleGroundBracketId bgBracketId = sBattleGroundMgr.GetBattleGroundBracketIdFromLevel(bgTypeId, _player->GetLevel()); // check queue conditions if (!joinAsGroup) @@ -156,13 +165,13 @@ void WorldSession::HandleBattlemasterJoinOpcode(WorldPacket& recv_data) } else { - grp = _player->GetGroup(); + group = _player->GetGroup(); // no group found, error - if (!grp) + if (!group) return; - uint32 err = grp->CanJoinBattleGroundQueue(bgTypeId, bgQueueTypeId, 0, bg->GetMaxPlayersPerTeam()); + uint32 err = group->CanJoinBattleGroundQueue(bgTypeId, bgQueueTypeId, 0, bg->GetMaxPlayersPerTeam()); isPremade = sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_PREMADE_GROUP_WAIT_FOR_MATCH) && - (grp->GetMembersCount() >= bg->GetMinPlayersPerTeam()); + (group->GetMembersCount() >= bg->GetMinPlayersPerTeam()); if (err != BG_JOIN_ERR_OK) { SendBattleGroundJoinError(err); @@ -171,51 +180,77 @@ void WorldSession::HandleBattlemasterJoinOpcode(WorldPacket& recv_data) } // if we're here, then the conditions to join a bg are met. We can proceed in joining. - // _player->GetGroup() was already checked, grp is already initialized - BattleGroundQueue& bgQueue = sBattleGroundMgr.m_battleGroundQueues[bgQueueTypeId]; + // _player->GetGroup() was already checked, group is already initialized + AddGroupToQueueInfo info; + info.team = _player->GetTeam(); + info.clientInstanceId = instanceId; + info.mapId = mapId; if (joinAsGroup) { + for (GroupReference* itr = group->GetFirstMember(); itr != nullptr; itr = itr->next()) + if (Player* member = itr->getSource()) + info.members.push_back(member->GetObjectGuid()); DEBUG_LOG("Battleground: the following players are joining as group:"); - GroupQueueInfo* queueInfo = bgQueue.AddGroup(_player, grp, bgTypeId, bgBracketId, isPremade, instanceId); - uint32 avgTime = bgQueue.GetAverageQueueWaitTime(queueInfo, _player->GetBattleGroundBracketIdFromLevel(bgTypeId)); - for (GroupReference* itr = grp->GetFirstMember(); itr != nullptr; itr = itr->next()) + sWorld.GetBGQueue().GetMessager().AddMessage([bgQueueTypeId, leaderGuid = group->GetLeaderGuid(), info, bgTypeId, bgBracketId, isPremade, instanceId, mapId](BattleGroundQueue* queue) { - Player* member = itr->getSource(); - if (!member) - continue; // this should never happen + BattleGroundQueueItem& queueItem = queue->GetBattleGroundQueue(bgQueueTypeId); + GroupQueueInfo* groupInfo = queueItem.AddGroup(leaderGuid, info, bgTypeId, bgBracketId, isPremade, instanceId); + uint32 avgTime = queueItem.GetAverageQueueWaitTime(groupInfo, bgBracketId); - uint32 queueSlot = member->AddBattleGroundQueueId(bgQueueTypeId); // add to queue - - // store entry point coords (same as leader entry point) - member->SetBattleGroundEntryPoint(_player); + sWorld.GetMessager().AddMessage([leaderGuid, members = info.members, bgQueueTypeId, bgTypeId, bgClientInstanceId = instanceId, avgTime, mapId](World* world) + { + Player* leader = sObjectMgr.GetPlayer(leaderGuid); + for (ObjectGuid guid : members) + { + Player* member = sObjectMgr.GetPlayer(guid); + if (!member) + continue; + + uint32 queueSlot = member->AddBattleGroundQueueId(bgQueueTypeId); // add to queue + + // store entry point coords (same as leader entry point) + member->SetBattleGroundEntryPoint(leader); + + // send status packet (in queue) + WorldPacket data; + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, bgTypeId, bgClientInstanceId, mapId, queueSlot, STATUS_WAIT_QUEUE, avgTime, 0); + member->GetSession()->SendPacket(data); + sBattleGroundMgr.BuildGroupJoinedBattlegroundPacket(data, bgTypeId, BG_GROUP_JOIN_STATUS_SUCCESS); + member->GetSession()->SendPacket(data); + DEBUG_LOG("Battleground: player joined queue for bg queue type %u bg type %u: GUID %u, NAME %s", bgQueueTypeId, bgTypeId, member->GetGUIDLow(), member->GetName()); + } - WorldPacket data; - // send status packet (in queue) - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, queueSlot, STATUS_WAIT_QUEUE, avgTime, 0); - member->GetSession()->SendPacket(data); - sBattleGroundMgr.BuildGroupJoinedBattlegroundPacket(data, bgTypeId, BG_GROUP_JOIN_STATUS_SUCCESS); - member->GetSession()->SendPacket(data); - DEBUG_LOG("Battleground: player joined queue for bg queue type %u bg type %u: GUID %u, NAME %s", bgQueueTypeId, bgTypeId, member->GetGUIDLow(), member->GetName()); - } - DEBUG_LOG("Battleground: group end"); + DEBUG_LOG("Battleground: group end"); + }); + queue->ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, bgBracketId); + }); } else { - GroupQueueInfo* queueInfo = bgQueue.AddGroup(_player, nullptr, bgTypeId, bgBracketId, isPremade, instanceId); - uint32 avgTime = bgQueue.GetAverageQueueWaitTime(queueInfo, _player->GetBattleGroundBracketIdFromLevel(bgTypeId)); - // already checked if queueSlot is valid, now just get it - uint32 queueSlot = _player->AddBattleGroundQueueId(bgQueueTypeId); - // store entry point coords - _player->SetBattleGroundEntryPoint(); - - WorldPacket data; - // send status packet (in queue) - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, queueSlot, STATUS_WAIT_QUEUE, avgTime, 0); - SendPacket(data); - DEBUG_LOG("Battleground: player joined queue for bg queue type %u bg type %u: GUID %u, NAME %s", bgQueueTypeId, bgTypeId, _player->GetGUIDLow(), _player->GetName()); + sWorld.GetBGQueue().GetMessager().AddMessage([bgQueueTypeId, playerGuid = _player->GetObjectGuid(), info, bgTypeId, bgBracketId, isPremade, instanceId, mapId](BattleGroundQueue* queue) + { + BattleGroundQueueItem& queueItem = queue->GetBattleGroundQueue(bgQueueTypeId); + GroupQueueInfo* groupInfo = queueItem.AddGroup(playerGuid, info, bgTypeId, bgBracketId, isPremade, instanceId); + uint32 avgTime = queueItem.GetAverageQueueWaitTime(groupInfo, bgBracketId); + sWorld.GetMessager().AddMessage([playerGuid, bgQueueTypeId, bgTypeId, bgClientInstanceId = instanceId, avgTime, mapId](World* world) + { + if (Player* player = sObjectMgr.GetPlayer(playerGuid)) + { + // already checked if queueSlot is valid, now just get it + uint32 queueSlot = player->AddBattleGroundQueueId(bgQueueTypeId); + // store entry point coords + player->SetBattleGroundEntryPoint(); + + WorldPacket data; + // send status packet (in queue) + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, bgTypeId, bgClientInstanceId, mapId, queueSlot, STATUS_WAIT_QUEUE, avgTime, 0); + player->GetSession()->SendPacket(data); + DEBUG_LOG("Battleground: player joined queue for bg queue type %u bg type %u: GUID %u, NAME %s", bgQueueTypeId, bgTypeId, player->GetGUIDLow(), player->GetName()); + } + }); + queue->ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, bgBracketId); + }); } - - sBattleGroundMgr.ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, _player->GetBattleGroundBracketIdFromLevel(bgTypeId)); } // Sent by client while inside battleground; depends on the battleground type @@ -314,10 +349,17 @@ void WorldSession::HandleBattlefieldListOpcode(WorldPacket& recv_data) sLog.outError("Battleground: invalid bgtype received."); return; } - - WorldPacket data; - sBattleGroundMgr.BuildBattleGroundListPacket(data, _player->GetObjectGuid(), _player, BattleGroundTypeId(bgTypeId)); - SendPacket(data); + + sWorld.GetBGQueue().GetMessager().AddMessage([playerGuid = _player->GetObjectGuid(), masterGuid = _player->GetObjectGuid(), playerLevel = _player->GetLevel(), bgTypeId](BattleGroundQueue* queue) + { + WorldPacket data; + queue->BuildBattleGroundListPacket(data, masterGuid, playerLevel, BattleGroundTypeId(bgTypeId)); + sWorld.GetMessager().AddMessage([playerGuid, data](World* world) + { + if (Player* player = sObjectMgr.GetPlayer(playerGuid)) + player->GetSession()->SendPacket(data); + }); + }); } // Sent by client when requesting teleport to the battleground location @@ -345,119 +387,124 @@ void WorldSession::HandleBattlefieldPortOpcode(WorldPacket& recv_data) // get GroupQueueInfo from BattleGroundQueue BattleGroundQueueTypeId bgQueueTypeId = BattleGroundMgr::BgQueueTypeId(bgTypeId); - BattleGroundQueue& bgQueue = sBattleGroundMgr.m_battleGroundQueues[bgQueueTypeId]; - - // we must use temporary variable, because GroupQueueInfo pointer can be deleted in BattleGroundQueue::RemovePlayer() function - GroupQueueInfo queueInfo; - if (!bgQueue.GetPlayerGroupInfoData(_player->GetObjectGuid(), &queueInfo)) - { - sLog.outError("BattlegroundHandler: itrplayerstatus not found."); - return; - } - - // if action == 1, then instanceId is required - if (!queueInfo.isInvitedToBgInstanceGuid && action == 1) - { - sLog.outError("BattlegroundHandler: instance not found."); - return; - } - - BattleGround* bg = sBattleGroundMgr.GetBattleGround(queueInfo.isInvitedToBgInstanceGuid, bgTypeId); - - // bg template might and must be used in case of leaving queue, when instance is not created yet - if (!bg && action == 0) - bg = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); - if (!bg) - { - sLog.outError("BattlegroundHandler: bg_template not found for type id %u.", bgTypeId); - return; - } + bool canJoinToBg = _player->CanJoinToBattleground(); + uint32 queueSlot = _player->GetBattleGroundQueueIndex(bgQueueTypeId); - // some checks if player isn't cheating - it is not exactly cheating, but we cannot allow it - if (action == 1) + sWorld.GetBGQueue().GetMessager().AddMessage([bgQueueTypeId, playerGuid = _player->GetObjectGuid(), actionTemp = action, canJoinToBg, bgTypeId, playerLevel = _player->GetLevel(), queueSlot](BattleGroundQueue* queue) { - // if player is trying to enter battleground and he has deserter debuff, we must just remove him from queue - if (!_player->CanJoinToBattleground()) + uint8 action = actionTemp; + BattleGroundQueueItem& queueItem = queue->GetBattleGroundQueue(bgQueueTypeId); + GroupQueueInfo queueInfo; + if (!queueItem.GetPlayerGroupInfoData(playerGuid, &queueInfo)) { - // send bg command result to show nice message - WorldPacket data2; - sBattleGroundMgr.BuildGroupJoinedBattlegroundPacket(data2, bgTypeId, BG_GROUP_JOIN_STATUS_DESERTERS); - _player->GetSession()->SendPacket(data2); - action = 0; - - DEBUG_LOG("Battleground: player %s (%u) has a deserter debuff, do not port him to battleground!", _player->GetName(), _player->GetGUIDLow()); + sLog.outError("BattlegroundHandler: itrplayerstatus not found."); + return; } - // if player don't match battleground max level, then do not allow him to enter! (this might happen when player leveled up during his waiting in queue - if (_player->GetLevel() > bg->GetMaxLevel()) + + if (!queueInfo.isInvitedToBgInstanceGuid && action == 1) { - sLog.outError("Battleground: Player %s (%u) has level (%u) higher than maxlevel (%u) of battleground (%u)! Do not port him to battleground!", - _player->GetName(), _player->GetGUIDLow(), _player->GetLevel(), bg->GetMaxLevel(), bg->GetTypeId()); - action = 0; + sLog.outError("BattlegroundHandler: instance not found."); + return; } - } - uint32 queueSlot = _player->GetBattleGroundQueueIndex(bgQueueTypeId); - WorldPacket data; + // some checks if player isn't cheating - it is not exactly cheating, but we cannot allow it + if (action == 1) + { + // if player is trying to enter battleground (not arena!) and he has deserter debuff, we must just remove him from queue + if (!canJoinToBg) + { + // send bg command result to show nice message + WorldPacket data2; + sBattleGroundMgr.BuildGroupJoinedBattlegroundPacket(data2, bgTypeId, BG_GROUP_JOIN_STATUS_DESERTERS); + sWorld.GetMessager().AddMessage([playerGuid, data2](World* world) + { + if (Player* player = sObjectMgr.GetPlayer(playerGuid)) + { + player->GetSession()->SendPacket(data2); + } + }); - switch (action) - { - case 1: // port to battleground - if (!_player->IsInvitedForBattleGroundQueueType(bgQueueTypeId)) - return; // cheating? + action = 0; + } + } - // resurrect the player - if (!_player->IsAlive()) + switch (action) + { + case 1: // port to battleground { - _player->ResurrectPlayer(1.0f); - _player->SpawnCorpseBones(); - } + BattleGroundInQueueInfo* bgInQueue = queue->GetFreeSlotInstance(bgTypeId, queueInfo.isInvitedToBgInstanceGuid); + MANGOS_ASSERT(bgInQueue); // at this point must always exist - _player->TaxiFlightInterrupt(); + // remove battleground queue status from BGmgr + queueItem.RemovePlayer(*queue, playerGuid, false); - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, queueSlot, STATUS_IN_PROGRESS, 0, bg->GetStartTime()); - _player->GetSession()->SendPacket(data); + sWorld.GetMessager().AddMessage([playerGuid, invitedTo = queueInfo.isInvitedToBgInstanceGuid, bgTypeId, bgQueueTypeId, groupTeam = queueInfo.groupTeam, queueSlot, bgClientInstanceId = bgInQueue->GetClientInstanceId(), mapId = bgInQueue->GetMapId()](World* world) + { + Player* player = sObjectMgr.GetPlayer(playerGuid); + if (!player) + return; - // remove battleground queue status from BGmgr - bgQueue.RemovePlayer(_player->GetObjectGuid(), false); + // resurrect the player + if (!player->IsAlive()) + { + player->ResurrectPlayer(1.0f); + player->SpawnCorpseBones(); + } - // this is still needed here if battleground "jumping" shouldn't add deserter debuff - // also this is required to prevent stuck at old battleground after SetBattleGroundId set to new - if (BattleGround* currentBg = _player->GetBattleGround()) - currentBg->RemovePlayerAtLeave(_player->GetObjectGuid(), false, true); + player->TaxiFlightInterrupt(); - // set the destination instance id - _player->SetBattleGroundId(bg->GetInstanceId(), bgTypeId); - // set the destination team - _player->SetBGTeam(queueInfo.groupTeam); - // bg->HandleBeforeTeleportToBattleGround(_player); - sBattleGroundMgr.SendToBattleGround(_player, queueInfo.isInvitedToBgInstanceGuid, bgTypeId); - // add only in HandleMoveWorldPortAck() - // bg->AddPlayer(_player,team); + uint32 startTime = 0; + if (BattleGround* bg = sBattleGroundMgr.GetBattleGround(invitedTo, bgTypeId)) + startTime = bg->GetStartTime(); - DEBUG_LOG("Battleground: player %s (%u) joined battle for bg %u, bgtype %u, queue type %u.", _player->GetName(), _player->GetGUIDLow(), bg->GetInstanceId(), bg->GetTypeId(), bgQueueTypeId); + WorldPacket data; + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, bgTypeId, bgClientInstanceId, mapId, queueSlot, STATUS_IN_PROGRESS, 0, startTime); + player->GetSession()->SendPacket(data); - break; - case 0: // leave queue - _player->RemoveBattleGroundQueueId(bgQueueTypeId); // must be called this way, because if you move this call to queue->removeplayer, it causes bugs - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, queueSlot, STATUS_NONE, 0, 0); - bgQueue.RemovePlayer(_player->GetObjectGuid(), true); - // player left queue, we should update it - sBattleGroundMgr.ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, _player->GetBattleGroundBracketIdFromLevel(bgTypeId)); - SendPacket(data); + // this is still needed here if battleground "jumping" shouldn't add deserter debuff + // also this is required to prevent stuck at old battleground after SetBattleGroundId set to new + if (BattleGround* currentBg = player->GetBattleGround()) + currentBg->RemovePlayerAtLeave(player->GetObjectGuid(), false, true); - DEBUG_LOG("Battleground: player %s (%u) left queue for bgtype %u, queue type %u.", _player->GetName(), _player->GetGUIDLow(), bg->GetTypeId(), bgQueueTypeId); + // set the destination instance id + player->SetBattleGroundId(invitedTo, bgTypeId); + // set the destination team + player->SetBGTeam(groupTeam); - break; - default: - sLog.outError("Battleground port: unknown action %u", action); - break; - } + sBattleGroundMgr.SendToBattleGround(player, invitedTo, bgTypeId); + + DEBUG_LOG("Battleground: player %s (%u) joined battle for bg %u, bgtype %u, queue type %u.", player->GetName(), player->GetGUIDLow(), invitedTo, bgTypeId, bgQueueTypeId); + }); + + break; + } + case 0: // leave queue + queueItem.RemovePlayer(*queue, playerGuid, true); + sWorld.GetMessager().AddMessage([playerGuid, bgQueueTypeId, queueSlot, bgTypeId, bgClientInstanceId = queueInfo.clientInstanceId, mapId = queueInfo.mapId](World* world) + { + Player* player = sObjectMgr.GetPlayer(playerGuid); + if (!player) + return; + player->RemoveBattleGroundQueueId(bgQueueTypeId); + WorldPacket data; + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, bgTypeId, bgClientInstanceId, mapId, queueSlot, STATUS_NONE, 0, 0); + player->GetSession()->SendPacket(data); + }); + + queue->ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, queueInfo.bgBracketId); + break; + default: + sLog.outError("Battleground port: unknown action %u", action); + break; + } + }); } // Sent by client when leaving the battleground void WorldSession::HandleLeaveBattlefieldOpcode(WorldPacket& recv_data) { DEBUG_LOG("WORLD: Received opcode CMSG_LEAVE_BATTLEFIELD"); + uint64 guid; uint32 mapId; recv_data >> mapId; @@ -482,6 +529,7 @@ void WorldSession::HandleBattlefieldStatusOpcode(WorldPacket& /*recv_data*/) WorldPacket data; // we must update all queues here BattleGround* bg; + std::vector> idsToCheck; for (uint8 i = 0; i < PLAYER_MAX_BATTLEGROUND_QUEUES; ++i) { BattleGroundQueueTypeId bgQueueTypeId = _player->GetBattleGroundQueueTypeId(i); @@ -497,41 +545,48 @@ void WorldSession::HandleBattlefieldStatusOpcode(WorldPacket& /*recv_data*/) { // this line is checked, i only don't know if GetStartTime is changing itself after bg end! // send status in BattleGround - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, i, STATUS_IN_PROGRESS, bg->GetEndTime(), bg->GetStartTime()); + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, bg->GetTypeId(), bg->GetClientInstanceId(), bg->GetMapId(), i, STATUS_IN_PROGRESS, bg->GetEndTime(), bg->GetStartTime()); SendPacket(data); continue; } } + + idsToCheck.push_back({i, bgQueueTypeId}); // we are sending update to player about queue - he can be invited there! - // get GroupQueueInfo for queue status - BattleGroundQueue& bgQueue = sBattleGroundMgr.m_battleGroundQueues[bgQueueTypeId]; - GroupQueueInfo queueInfo; - if (!bgQueue.GetPlayerGroupInfoData(_player->GetObjectGuid(), &queueInfo)) - continue; + } - if (queueInfo.isInvitedToBgInstanceGuid) + sWorld.GetBGQueue().GetMessager().AddMessage([idsToCheck, playerGuid = _player->GetObjectGuid(), playerLevel = _player->GetLevel()](BattleGroundQueue* queue) + { + for (auto [queueSlot, bgQueueTypeId] : idsToCheck) { - bg = sBattleGroundMgr.GetBattleGround(queueInfo.isInvitedToBgInstanceGuid, bgTypeId); - if (!bg) + BattleGroundQueueItem& queueItem = queue->GetBattleGroundQueue(bgQueueTypeId); + GroupQueueInfo queueInfo; + if (!queueItem.GetPlayerGroupInfoData(playerGuid, &queueInfo)) continue; - uint32 remainingTime = WorldTimer::getMSTimeDiff(WorldTimer::getMSTime(), queueInfo.removeInviteTime); - // send status invited to BattleGround - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, i, STATUS_WAIT_JOIN, remainingTime, 0); - SendPacket(data); + WorldPacket data; + if (queueInfo.isInvitedToBgInstanceGuid) + { + uint32 remainingTime = WorldTimer::getMSTimeDiff(WorldTimer::getMSTime(), queueInfo.removeInviteTime); + // send status invited to BattleGround + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, queueInfo.bgTypeId, queueInfo.clientInstanceId, queueInfo.mapId, queueSlot, STATUS_WAIT_JOIN, remainingTime, 0); + } + else + { + uint32 avgTime = queueItem.GetAverageQueueWaitTime(&queueInfo, sBattleGroundMgr.GetBattleGroundBracketIdFromLevel(queueInfo.bgTypeId, playerLevel)); + // send status in BattleGround Queue + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, true, queueInfo.bgTypeId, queueInfo.clientInstanceId, queueInfo.mapId, queueSlot, STATUS_WAIT_QUEUE, avgTime, WorldTimer::getMSTimeDiff(queueInfo.joinTime, WorldTimer::getMSTime())); + } + sWorld.GetMessager().AddMessage([playerGuid, data](World* world) + { + if (Player* player = sObjectMgr.GetPlayer(playerGuid)) + { + player->GetSession()->SendPacket(data); + } + }); } - else - { - bg = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); - if (!bg) - continue; + }); - uint32 avgTime = bgQueue.GetAverageQueueWaitTime(&queueInfo, _player->GetBattleGroundBracketIdFromLevel(bgTypeId)); - // send status in BattleGround Queue - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, i, STATUS_WAIT_QUEUE, avgTime, WorldTimer::getMSTimeDiff(queueInfo.joinTime, WorldTimer::getMSTime())); - SendPacket(data); - } - } } // Sent by client when requesting the spirit healer diff --git a/src/game/BattleGround/BattleGroundMgr.cpp b/src/game/BattleGround/BattleGroundMgr.cpp index 230bc6c9fb..48c1ab6a86 100644 --- a/src/game/BattleGround/BattleGroundMgr.cpp +++ b/src/game/BattleGround/BattleGroundMgr.cpp @@ -37,884 +37,14 @@ INSTANTIATE_SINGLETON_1(BattleGroundMgr); -/*********************************************************/ -/*** BATTLEGROUND QUEUE SYSTEM ***/ -/*********************************************************/ - -BattleGroundQueue::BattleGroundQueue() -{ - for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) - { - for (uint8 j = 0; j < MAX_BATTLEGROUND_BRACKETS; ++j) - { - m_sumOfWaitTimes[i][j] = 0; - m_waitTimeLastPlayer[i][j] = 0; - - for (uint8 k = 0; k < COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME; ++k) - m_waitTimes[i][j][k] = 0; - } - } -} - -BattleGroundQueue::~BattleGroundQueue() -{ - m_queuedPlayers.clear(); - for (auto& m_queuedGroup : m_queuedGroups) - { - for (uint8 j = 0; j < BG_QUEUE_GROUP_TYPES_COUNT; ++j) - { - for (GroupsQueueType::iterator itr = m_queuedGroup[j].begin(); itr != m_queuedGroup[j].end(); ++itr) - delete (*itr); - - m_queuedGroup[j].clear(); - } - } -} - -/*********************************************************/ -/*** BATTLEGROUND QUEUE SELECTION POOLS ***/ -/*********************************************************/ - -// selection pool initialization, used to clean up from prev selection -void BattleGroundQueue::SelectionPool::Init() -{ - selectedGroups.clear(); - playerCount = 0; -} - -/** - Function that removes group infr from pool selection - - returns true when we need to try to add new group to selection pool - - returns false when selection pool is ok or when we kicked smaller group than we need to kick - - sometimes it can be called on empty selection pool - - @param size -*/ -bool BattleGroundQueue::SelectionPool::KickGroup(uint32 size) -{ - // find maxgroup or LAST group with size == size and kick it - bool found = false; - GroupsQueueType::iterator groupToKick = selectedGroups.begin(); - - for (GroupsQueueType::iterator itr = groupToKick; itr != selectedGroups.end(); ++itr) - { - if (abs((int32)((*itr)->players.size() - size)) <= 1) - { - groupToKick = itr; - found = true; - } - else if (!found && (*itr)->players.size() >= (*groupToKick)->players.size()) - groupToKick = itr; - } - - // if pool is empty, do nothing - if (GetPlayerCount()) - { - // update player count - GroupQueueInfo* queueInfo = (*groupToKick); - selectedGroups.erase(groupToKick); - playerCount -= queueInfo->players.size(); - - // return false if we kicked smaller group or there are enough players in selection pool - if (queueInfo->players.size() <= size + 1) - return false; - } - return true; -} - -/** - Function that adds group to selection pool - - returns true if we can invite more players, or when we added group to selection pool - - returns false when selection pool is full - - @param group queue info - @param desired count -*/ -bool BattleGroundQueue::SelectionPool::AddGroup(GroupQueueInfo* queueInfo, uint32 desiredCount, uint32 bgInstanceId) -{ - // if group is larger than desired count - don't allow to add it to pool - if (!queueInfo->isInvitedToBgInstanceGuid && - (!queueInfo->desiredInstanceId || queueInfo->desiredInstanceId == bgInstanceId) && - (desiredCount >= playerCount + queueInfo->players.size())) - { - selectedGroups.push_back(queueInfo); - // increase selected players count - playerCount += queueInfo->players.size(); - - return true; - } - - return playerCount < desiredCount; -} - -/*********************************************************/ -/*** BATTLEGROUND QUEUES ***/ -/*********************************************************/ - -// add group or player (grp == nullptr) to bg queue with the given leader and bg specifications -GroupQueueInfo* BattleGroundQueue::AddGroup(Player* leader, Group* group, BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId, bool isPremade, uint32 instanceId) -{ - // create new ginfo - GroupQueueInfo* queueInfo = new GroupQueueInfo; - queueInfo->bgTypeId = bgTypeId; - queueInfo->isInvitedToBgInstanceGuid = 0; - queueInfo->joinTime = WorldTimer::getMSTime(); - queueInfo->removeInviteTime = 0; - queueInfo->groupTeam = leader->GetTeam(); - queueInfo->desiredInstanceId = instanceId; - - queueInfo->players.clear(); - - // compute index (if group is premade or joined a rated match) to queues - uint32 index = 0; - if (!isPremade) - index += PVP_TEAM_COUNT; // BG_QUEUE_PREMADE_* -> BG_QUEUE_NORMAL_* - - if (queueInfo->groupTeam == HORDE) - ++index; // BG_QUEUE_*_ALLIANCE -> BG_QUEUE_*_HORDE - - DEBUG_LOG("Adding Group to BattleGroundQueue bgTypeId : %u, bracket_id : %u, index : %u", bgTypeId, bracketId, index); - - uint32 lastOnlineTime = WorldTimer::getMSTime(); - - // add players from group to ginfo - { - // std::lock_guard guard(m_Lock); - if (group) - { - for (GroupReference* itr = group->GetFirstMember(); itr != nullptr; itr = itr->next()) - { - Player* member = itr->getSource(); - if (!member) - continue; // this should never happen - PlayerQueueInfo& playerInfo = m_queuedPlayers[member->GetObjectGuid()]; - playerInfo.lastOnlineTime = lastOnlineTime; - playerInfo.groupInfo = queueInfo; - // add the pinfo to ginfo's list - queueInfo->players[member->GetObjectGuid()] = &playerInfo; - } - } - else - { - PlayerQueueInfo& playerInfo = m_queuedPlayers[leader->GetObjectGuid()]; - playerInfo.lastOnlineTime = lastOnlineTime; - playerInfo.groupInfo = queueInfo; - queueInfo->players[leader->GetObjectGuid()] = &playerInfo; - } - - // add GroupInfo to m_QueuedGroups - m_queuedGroups[bracketId][index].push_back(queueInfo); - - // announce to world, this code needs mutex - if (!isPremade && sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_QUEUE_ANNOUNCER_JOIN)) - { - if (BattleGround* bg = sBattleGroundMgr.GetBattleGroundTemplate(queueInfo->bgTypeId)) - { - char const* bgName = bg->GetName(); - uint32 minPlayers = bg->GetMinPlayersPerTeam(); - uint32 qHorde = 0; - uint32 qAlliance = 0; - uint32 q_min_level = leader->GetMinLevelForBattleGroundBracketId(bracketId, bgTypeId); - GroupsQueueType::const_iterator itr; - for (itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].begin(); itr != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].end(); ++itr) - if (!(*itr)->isInvitedToBgInstanceGuid) - qAlliance += (*itr)->players.size(); - - for (itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].begin(); itr != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].end(); ++itr) - if (!(*itr)->isInvitedToBgInstanceGuid) - qHorde += (*itr)->players.size(); - - // Show queue status to player only (when joining queue) - if (sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_QUEUE_ANNOUNCER_JOIN) == 1) - { - ChatHandler(leader).PSendSysMessage(LANG_BG_QUEUE_ANNOUNCE_SELF, bgName, q_min_level, q_min_level + 10, - qAlliance, (minPlayers > qAlliance) ? minPlayers - qAlliance : (uint32)0, qHorde, (minPlayers > qHorde) ? minPlayers - qHorde : (uint32)0); - } - // System message - else - { - sWorld.SendWorldText(LANG_BG_QUEUE_ANNOUNCE_WORLD, bgName, q_min_level, q_min_level + 10, - qAlliance, (minPlayers > qAlliance) ? minPlayers - qAlliance : (uint32)0, qHorde, (minPlayers > qHorde) ? minPlayers - qHorde : (uint32)0); - } - } - } - // release mutex - } - - return queueInfo; -} - -/** - Method that updates average update wait time - - @param group queue info - @param bracket id -*/ -void BattleGroundQueue::PlayerInvitedToBgUpdateAverageWaitTime(GroupQueueInfo* queueInfo, BattleGroundBracketId bracketId) -{ - uint32 timeInQueue = WorldTimer::getMSTimeDiff(queueInfo->joinTime, WorldTimer::getMSTime()); - uint8 teamIndex = TEAM_INDEX_ALLIANCE; // default set to BG_TEAM_ALLIANCE - or non rated arenas! - - if (queueInfo->groupTeam == HORDE) - teamIndex = TEAM_INDEX_HORDE; - - // store pointer to arrayindex of player that was added first - uint32* lastPlayerAddedPointer = &(m_waitTimeLastPlayer[teamIndex][bracketId]); - - // remove his time from sum - m_sumOfWaitTimes[teamIndex][bracketId] -= m_waitTimes[teamIndex][bracketId][(*lastPlayerAddedPointer)]; - - // set average time to new - m_waitTimes[teamIndex][bracketId][(*lastPlayerAddedPointer)] = timeInQueue; - - // add new time to sum - m_sumOfWaitTimes[teamIndex][bracketId] += timeInQueue; - - // set index of last player added to next one - (*lastPlayerAddedPointer)++; - (*lastPlayerAddedPointer) %= COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME; -} - -/** - Function that returns averate queue wait time - - @param group queue info - @param bracket id -*/ -uint32 BattleGroundQueue::GetAverageQueueWaitTime(GroupQueueInfo* queueInfo, BattleGroundBracketId bracketId) -{ - uint8 teamIndex = TEAM_INDEX_ALLIANCE; // default set to BG_TEAM_ALLIANCE - or non rated arenas! - if (queueInfo->groupTeam == HORDE) - teamIndex = TEAM_INDEX_HORDE; - - // check if there is enought values(we always add values > 0) - if (m_waitTimes[teamIndex][bracketId][COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME - 1]) - return (m_sumOfWaitTimes[teamIndex][bracketId] / COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME); - - // if there aren't enough values return 0 - not available - return 0; -} - -/** - Method that removes player from queue and from group info, if group info is empty then remove it too - - @param guid - @param decrease invite count -*/ -void BattleGroundQueue::RemovePlayer(ObjectGuid guid, bool decreaseInvitedCount) -{ - // Player *plr = sObjectMgr.GetPlayer(guid); - // std::lock_guard guard(m_Lock); - - int32 bracketId = -1; // signed for proper for-loop finish - - // remove player from map, if he's there - QueuedPlayersMap::iterator itr = m_queuedPlayers.find(guid); - if (itr == m_queuedPlayers.end()) - { - sLog.outError("BattleGroundQueue: couldn't find for remove: %s", guid.GetString().c_str()); - return; - } - - GroupQueueInfo* group = itr->second.groupInfo; - GroupsQueueType::iterator group_itr; - // mostly people with the highest levels are in battlegrounds, thats why - // we count from MAX_BATTLEGROUND_QUEUES - 1 to 0 - // variable index removes useless searching in other team's queue - uint32 index = GetTeamIndexByTeamId(group->groupTeam); - - for (int8 bracketIdTmp = MAX_BATTLEGROUND_BRACKETS - 1; bracketIdTmp >= 0 && bracketId == -1; --bracketIdTmp) - { - // we must check premade and normal team's queue - because when players from premade are joining bg, - // they leave groupinfo so we can't use its players size to find out index - for (uint8 j = index; j < BG_QUEUE_GROUP_TYPES_COUNT; j += BG_QUEUE_NORMAL_ALLIANCE) - { - for (GroupsQueueType::iterator group_itr_tmp = m_queuedGroups[bracketIdTmp][j].begin(); group_itr_tmp != m_queuedGroups[bracketIdTmp][j].end(); ++group_itr_tmp) - { - if ((*group_itr_tmp) == group) - { - bracketId = bracketIdTmp; - group_itr = group_itr_tmp; - // we must store index to be able to erase iterator - index = j; - break; - } - } - } - } - - // player can't be in queue without group, but just in case - if (bracketId == -1) - { - sLog.outError("BattleGroundQueue: ERROR Cannot find groupinfo for %s", guid.GetString().c_str()); - return; - } - DEBUG_LOG("BattleGroundQueue: Removing %s, from bracket_id %u", guid.GetString().c_str(), (uint32)bracketId); - - // ALL variables are correctly set - // We can ignore leveling up in queue - it should not cause crash - // remove player from group - // if only one player there, remove group - - // remove player queue info from group queue info - GroupQueueInfoPlayers::iterator pitr = group->players.find(guid); - if (pitr != group->players.end()) - group->players.erase(pitr); - - // if invited to bg, and should decrease invited count, then do it - if (decreaseInvitedCount && group->isInvitedToBgInstanceGuid) - { - BattleGround* bg = sBattleGroundMgr.GetBattleGround(group->isInvitedToBgInstanceGuid, group->bgTypeId); - if (bg) - bg->DecreaseInvitedCount(group->groupTeam); - } - - // remove player queue info - m_queuedPlayers.erase(itr); - - // remove group queue info if needed - if (group->players.empty()) - { - m_queuedGroups[bracketId][index].erase(group_itr); - delete group; - } -} - -/** - Function that returns true when player is in queue and is invited to bgInstanceGuid - - @param player guid - @param battleground instance guid - @param remove time -*/ -bool BattleGroundQueue::IsPlayerInvited(ObjectGuid playerGuid, const uint32 bgInstanceGuid, const uint32 removeTime) -{ - // std::lock_guard g(m_Lock); - QueuedPlayersMap::const_iterator qItr = m_queuedPlayers.find(playerGuid); - return (qItr != m_queuedPlayers.end() - && qItr->second.groupInfo->isInvitedToBgInstanceGuid == bgInstanceGuid - && qItr->second.groupInfo->removeInviteTime == removeTime); -} - -/** - Function that returns player group info data - - returns true when the player is found in queue - - @param player guid - @param group queue info -*/ -bool BattleGroundQueue::GetPlayerGroupInfoData(ObjectGuid guid, GroupQueueInfo* queueInfo) -{ - // std::lock_guard g(m_Lock); - QueuedPlayersMap::const_iterator qItr = m_queuedPlayers.find(guid); - if (qItr == m_queuedPlayers.end()) - return false; - - *queueInfo = *(qItr->second.groupInfo); - return true; -} - -/** - Function that invites group to battleground - - @param group queue info - @param battleground - @param team -*/ -bool BattleGroundQueue::InviteGroupToBg(GroupQueueInfo* queueInfo, BattleGround* bg, Team team) -{ - // set side if needed - if (team == ALLIANCE || team == HORDE) - queueInfo->groupTeam = team; - - if (!queueInfo->isInvitedToBgInstanceGuid) - { - // not yet invited - // set invitation - queueInfo->isInvitedToBgInstanceGuid = bg->GetInstanceId(); - BattleGroundTypeId bgTypeId = bg->GetTypeId(); - BattleGroundQueueTypeId bgQueueTypeId = BattleGroundMgr::BgQueueTypeId(bgTypeId); - BattleGroundBracketId bracket_id = bg->GetBracketId(); - - queueInfo->removeInviteTime = WorldTimer::getMSTime() + INVITE_ACCEPT_WAIT_TIME; - - // loop through the players - for (GroupQueueInfoPlayers::iterator itr = queueInfo->players.begin(); itr != queueInfo->players.end(); ++itr) - { - // get the player - Player* plr = sObjectMgr.GetPlayer(itr->first); - // if offline, skip him, this should not happen - player is removed from queue when he logs out - if (!plr) - continue; - - // invite the player - PlayerInvitedToBgUpdateAverageWaitTime(queueInfo, bracket_id); - // sBattleGroundMgr.InvitePlayer(plr, bg, ginfo->Team); - - // set invited player counters - bg->IncreaseInvitedCount(queueInfo->groupTeam); - - plr->SetInviteForBattleGroundQueueType(bgQueueTypeId, queueInfo->isInvitedToBgInstanceGuid); - - // create remind invite events - BgQueueInviteEvent* inviteEvent = new BgQueueInviteEvent(plr->GetObjectGuid(), queueInfo->isInvitedToBgInstanceGuid, bgTypeId, queueInfo->removeInviteTime); - plr->m_events.AddEvent(inviteEvent, plr->m_events.CalculateTime(INVITATION_REMIND_TIME)); - - WorldPacket data; - - uint32 queueSlot = plr->GetBattleGroundQueueIndex(bgQueueTypeId); - - DEBUG_LOG("Battleground: invited %s to BG instance %u queueindex %u bgtype %u, I can't help it if they don't press the enter battle button.", plr->GetGuidStr().c_str(), bg->GetInstanceId(), queueSlot, bg->GetTypeId()); - - // send status packet - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, queueSlot, STATUS_WAIT_JOIN, INVITE_ACCEPT_WAIT_TIME, 0); - plr->GetSession()->SendPacket(data); - } - return true; - } - - return false; -} - -/** - Method that invites players to an already running battleground - - invitation is based on config file - - large groups are disadvantageous, because they will be kicked first if invitation type = 1 - - @param battleground - @param bracket id -*/ -void BattleGroundQueue::FillPlayersToBg(BattleGround* bg, BattleGroundBracketId bracketId) -{ - int32 hordeFree = bg->GetFreeSlotsForTeam(HORDE); - int32 aliFree = bg->GetFreeSlotsForTeam(ALLIANCE); - - // iterator for iterating through bg queue - GroupsQueueType::const_iterator Ali_itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].begin(); - // count of groups in queue - used to stop cycles - uint32 aliCount = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].size(); - // index to queue which group is current - uint32 aliIndex = 0; - for (; aliIndex < aliCount && m_selectionPools[TEAM_INDEX_ALLIANCE].AddGroup((*Ali_itr), aliFree, bg->GetClientInstanceId()); ++aliIndex) - ++Ali_itr; - - // the same thing for horde - GroupsQueueType::const_iterator Horde_itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].begin(); - uint32 hordeCount = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].size(); - uint32 hordeIndex = 0; - for (; hordeIndex < hordeCount && m_selectionPools[TEAM_INDEX_HORDE].AddGroup((*Horde_itr), hordeFree, bg->GetClientInstanceId()); ++hordeIndex) - ++Horde_itr; - - // if ofc like BG queue invitation is set in config, then we are happy - if (sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_INVITATION_TYPE) == 0) - return; - - /* - if we reached this code, then we have to solve NP - complete problem called Subset sum problem - So one solution is to check all possible invitation subgroups, or we can use these conditions: - 1. Last time when BattleGroundQueue::Update was executed we invited all possible players - so there is only small possibility - that we will invite now whole queue, because only 1 change has been made to queues from the last BattleGroundQueue::Update call - 2. Other thing we should consider is group order in queue - */ - - // At first we need to compare free space in bg and our selection pool - int32 diffAli = aliFree - int32(m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()); - int32 diffHorde = hordeFree - int32(m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()); - while (abs(diffAli - diffHorde) > 1 && (m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() > 0 || m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() > 0)) - { - // each cycle execution we need to kick at least 1 group - if (diffAli < diffHorde) - { - // kick alliance group, add to pool new group if needed - if (m_selectionPools[TEAM_INDEX_ALLIANCE].KickGroup(diffHorde - diffAli)) - { - for (; aliIndex < aliCount && m_selectionPools[TEAM_INDEX_ALLIANCE].AddGroup((*Ali_itr), (aliFree >= diffHorde) ? aliFree - diffHorde : 0, bg->GetClientInstanceId()); ++aliIndex) - ++Ali_itr; - } - // if ali selection is already empty, then kick horde group, but if there are less horde than ali in bg - break; - if (!m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()) - { - if (aliFree <= diffHorde + 1) - break; - m_selectionPools[TEAM_INDEX_HORDE].KickGroup(diffHorde - diffAli); - } - } - else - { - // kick horde group, add to pool new group if needed - if (m_selectionPools[TEAM_INDEX_HORDE].KickGroup(diffAli - diffHorde)) - { - for (; hordeIndex < hordeCount && m_selectionPools[TEAM_INDEX_HORDE].AddGroup((*Horde_itr), (hordeFree >= diffAli) ? hordeFree - diffAli : 0, bg->GetClientInstanceId()); ++hordeIndex) - ++Horde_itr; - } - if (!m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()) - { - if (hordeFree <= diffAli + 1) - break; - m_selectionPools[TEAM_INDEX_ALLIANCE].KickGroup(diffAli - diffHorde); - } - } - - // count diffs after small update - diffAli = aliFree - int32(m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()); - diffHorde = hordeFree - int32(m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()); - } -} - -/** - Method that checks if (premade vs premade) battlegrouns is possible - - then after 30 mins (default) in queue it moves premade group to normal queue - - it tries to invite as much players as it can - to maxPlayersPerTeam, because premade groups have more than minPlayersPerTeam players - - @param bracket id - @param min players per team - @param max players per team -*/ -bool BattleGroundQueue::CheckPremadeMatch(BattleGroundBracketId bracketId, uint32 minPlayersPerTeam, uint32 maxPlayersPerTeam) -{ - // check match - if (!m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].empty() && !m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].empty()) - { - // start premade match - // if groups aren't invited - GroupsQueueType::const_iterator ali_group, horde_group; - for (ali_group = m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].begin(); ali_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].end(); ++ali_group) - if (!(*ali_group)->isInvitedToBgInstanceGuid) - break; - - for (horde_group = m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].begin(); horde_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].end(); ++horde_group) - if (!(*horde_group)->isInvitedToBgInstanceGuid) - break; - - if (ali_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].end() && horde_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].end()) - { - m_selectionPools[TEAM_INDEX_ALLIANCE].AddGroup((*ali_group), maxPlayersPerTeam, 0); - m_selectionPools[TEAM_INDEX_HORDE].AddGroup((*horde_group), maxPlayersPerTeam, 0); - - // add groups/players from normal queue to size of bigger group - uint32 maxPlayers = std::max(m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount(), m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()); - for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) - { - for (GroupsQueueType::const_iterator itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].begin(); itr != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].end(); ++itr) - { - // if itr can join BG and player count is less that maxPlayers, then add group to selectionpool - if (!(*itr)->isInvitedToBgInstanceGuid && !m_selectionPools[i].AddGroup((*itr), maxPlayers, 0)) - break; - } - } - - // premade selection pools are set - return true; - } - } - // now check if we can move group from Premade queue to normal queue (timer has expired) or group size lowered!! - // this could be 2 cycles but i'm checking only first team in queue - it can cause problem - - // if first is invited to BG and seconds timer expired, but we can ignore it, because players have only 80 seconds to click to enter bg - // and when they click or after 80 seconds the queue info is removed from queue - uint32 time_before = WorldTimer::getMSTime() - sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_PREMADE_GROUP_WAIT_FOR_MATCH); - for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) - { - if (!m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE + i].empty()) - { - GroupsQueueType::iterator itr = m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE + i].begin(); - if (!(*itr)->isInvitedToBgInstanceGuid && ((*itr)->joinTime < time_before || (*itr)->players.size() < minPlayersPerTeam)) - { - // we must insert group to normal queue and erase pointer from premade queue - m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].push_front((*itr)); - m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE + i].erase(itr); - } - } - } - - // selection pools are not set - return false; -} - -/** - Method that tries to create battleground with minPlayersPerTeam against maxPlayersPerTeam - - @param battleground - @param bracket id - @param min players - @param max players -*/ -bool BattleGroundQueue::CheckNormalMatch(BattleGroundBracketId bracketId, uint32 minPlayers, uint32 maxPlayers) -{ - GroupsQueueType::const_iterator itr_team[PVP_TEAM_COUNT]; - for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) - { - itr_team[i] = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].begin(); - for (; itr_team[i] != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].end(); ++(itr_team[i])) - { - if (!(*(itr_team[i]))->isInvitedToBgInstanceGuid) - { - m_selectionPools[i].AddGroup(*(itr_team[i]), maxPlayers, 0); - if (m_selectionPools[i].GetPlayerCount() >= minPlayers) - break; - } - } - } - - // try to invite same number of players - this cycle may cause longer wait time even if there are enough players in queue, but we want ballanced bg - uint32 j = TEAM_INDEX_ALLIANCE; - if (m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() < m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()) - j = TEAM_INDEX_HORDE; - - if (sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_INVITATION_TYPE) != 0 - && m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() >= minPlayers && m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() >= minPlayers) - { - // we will try to invite more groups to team with less players indexed by j - ++(itr_team[j]); // this will not cause a crash, because for cycle above reached break; - for (; itr_team[j] != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + j].end(); ++(itr_team[j])) - { - if (!(*(itr_team[j]))->isInvitedToBgInstanceGuid) - if (!m_selectionPools[j].AddGroup(*(itr_team[j]), m_selectionPools[(j + 1) % PVP_TEAM_COUNT].GetPlayerCount(), 0)) - break; - } - // do not allow to start bg with more than 2 players more on 1 faction - if (abs((int32)(m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() - m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount())) > 2) - return false; - } - - // allow 1v0 if debug bg - if (sBattleGroundMgr.IsTesting() && (m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() || m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount())) - return true; - - // return true if there are enough players in selection pools - enable to work .debug bg command correctly - return m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() >= minPlayers && m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() >= minPlayers; -} - -/** - Method that is called when group is inserted, or player / group is removed from BG Queue - there is only one player's status changed, so we don't use while(true) cycles to invite whole queue - - it must be called after fully adding the members of a group to ensure group joining - - should be called from BattleGround::RemovePlayer function in some cases - - @param bg type id - @param bracket id - @param arena type - @param isRated - @param arenaRating -*/ -void BattleGroundQueue::Update(BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId) -{ - // std::lock_guard guard(m_Lock); - // if no players in queue - do nothing - if (m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].empty() && - m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].empty() && - m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].empty() && - m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].empty()) - return; - - // battleground with free slot for player should be always in the beggining of the queue - // maybe it would be better to create bgfreeslotqueue for each bracket_id - BgFreeSlotQueueType::iterator next; - for (BgFreeSlotQueueType::iterator itr = sBattleGroundMgr.BgFreeSlotQueue[bgTypeId].begin(); itr != sBattleGroundMgr.BgFreeSlotQueue[bgTypeId].end(); itr = next) - { - next = itr; - ++next; - // battleground is running, so if: - if ((*itr)->GetTypeId() == bgTypeId && (*itr)->GetBracketId() == bracketId && - (*itr)->GetStatus() > STATUS_WAIT_QUEUE && (*itr)->GetStatus() < STATUS_WAIT_LEAVE) - { - BattleGround* bg = *itr; // we have to store battleground pointer here, because when battleground is full, it is removed from free queue (not yet implemented!!) - // and iterator is invalid - - // clear selection pools - m_selectionPools[TEAM_INDEX_ALLIANCE].Init(); - m_selectionPools[TEAM_INDEX_HORDE].Init(); - - // call a function that does the job for us - FillPlayersToBg(bg, bracketId); - - // now everything is set, invite players - for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_ALLIANCE].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_ALLIANCE].selectedGroups.end(); ++citr) - InviteGroupToBg((*citr), bg, (*citr)->groupTeam); - for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_HORDE].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_HORDE].selectedGroups.end(); ++citr) - InviteGroupToBg((*citr), bg, (*citr)->groupTeam); - - if (!bg->HasFreeSlots()) - { - // remove BG from BGFreeSlotQueue - bg->RemoveFromBgFreeSlotQueue(); - } - } - } - - // finished iterating through the bgs with free slots, maybe we need to create a new bg - - BattleGround* bgTemplate = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); - if (!bgTemplate) - { - sLog.outError("Battleground: Update: bg template not found for %u", bgTypeId); - return; - } - - // get the min. players per team, properly for larger arenas as well. - uint32 minPlayersPerTeam = bgTemplate->GetMinPlayersPerTeam(); - uint32 maxPlayersPerTeam = bgTemplate->GetMaxPlayersPerTeam(); - if (sBattleGroundMgr.IsTesting()) - minPlayersPerTeam = 1; - - m_selectionPools[TEAM_INDEX_ALLIANCE].Init(); - m_selectionPools[TEAM_INDEX_HORDE].Init(); - - { - // check if there is premade against premade match - if (CheckPremadeMatch(bracketId, minPlayersPerTeam, maxPlayersPerTeam)) - { - // create new battleground - BattleGround* bg2 = sBattleGroundMgr.CreateNewBattleGround(bgTypeId, bracketId); - if (!bg2) - { - sLog.outError("BattleGroundQueue::Update - Cannot create battleground: %u", bgTypeId); - return; - } - - // invite those selection pools - for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) - for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.end(); ++citr) - InviteGroupToBg((*citr), bg2, (*citr)->groupTeam); - - // start bg - bg2->StartBattleGround(); - // clear structures - m_selectionPools[TEAM_INDEX_ALLIANCE].Init(); - m_selectionPools[TEAM_INDEX_HORDE].Init(); - } - } - - // now check if there are in queues enough players to start new game of (normal battleground) - { - // if there are enough players in pools, start new battleground or non rated arena - if (CheckNormalMatch(bracketId, minPlayersPerTeam, maxPlayersPerTeam)) - { - // we successfully created a pool - BattleGround* bg2 = sBattleGroundMgr.CreateNewBattleGround(bgTypeId, bracketId); - if (!bg2) - { - sLog.outError("BattleGroundQueue::Update - Cannot create battleground: %u", bgTypeId); - return; - } - - // invite those selection pools - for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) - for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.end(); ++citr) - InviteGroupToBg((*citr), bg2, (*citr)->groupTeam); - - // start bg - bg2->StartBattleGround(); - } - } -} - -/*********************************************************/ -/*** BATTLEGROUND QUEUE EVENTS ***/ -/*********************************************************/ - -/** - Function that executes battleground queue invite event -*/ -bool BgQueueInviteEvent::Execute(uint64 /*e_time*/, uint32 /*p_time*/) -{ - sWorld.GetMessager().AddMessage([event = *this](World* /*world*/) - { - Player* plr = sObjectMgr.GetPlayer(event.m_playerGuid); - // player logged off (we should do nothing, he is correctly removed from queue in another procedure) - if (!plr) - return; - - BattleGround* bg = sBattleGroundMgr.GetBattleGround(event.m_bgInstanceGuid, event.m_bgTypeId); - // if battleground ended and its instance deleted - do nothing - if (!bg) - return; - - BattleGroundQueueTypeId bgQueueTypeId = BattleGroundMgr::BgQueueTypeId(bg->GetTypeId()); - uint32 queueSlot = plr->GetBattleGroundQueueIndex(bgQueueTypeId); - if (queueSlot < PLAYER_MAX_BATTLEGROUND_QUEUES) // player is in queue or in battleground - { - // check if player is invited to this bg - BattleGroundQueue& bgQueue = sBattleGroundMgr.m_battleGroundQueues[bgQueueTypeId]; - if (bgQueue.IsPlayerInvited(event.m_playerGuid, event.m_bgInstanceGuid, event.m_removeTime)) - { - WorldPacket data; - // we must send remaining time in queue - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, queueSlot, STATUS_WAIT_JOIN, INVITE_ACCEPT_WAIT_TIME - INVITATION_REMIND_TIME, 0); - plr->GetSession()->SendPacket(data); - } - } - }); - return true; // event will be deleted -} - -void BgQueueInviteEvent::Abort(uint64 /*e_time*/) -{ - // do nothing -} - -/** - Function that executes battleground queue remove event - this event has many possibilities when it is executed: - 1. player is in battleground ( he clicked enter on invitation window ) - 2. player left battleground queue and he isn't there any more - 3. player left battleground queue and he joined it again and IsInvitedToBGInstanceGUID = 0 - 4. player left queue and he joined again and he has been invited to same battleground again -> we should not remove him from queue yet - 5. player is invited to bg and he didn't choose what to do and timer expired - only in this condition we should call queue::RemovePlayer - we must remove player in the 5. case even if battleground object doesn't exist! -*/ -bool BgQueueRemoveEvent::Execute(uint64 /*e_time*/, uint32 /*p_time*/) -{ - sWorld.GetMessager().AddMessage([event = *this](World* /*world*/) - { - Player* plr = sObjectMgr.GetPlayer(event.m_playerGuid); - if (!plr) - // player logged off (we should do nothing, he is correctly removed from queue in another procedure) - return; - - BattleGround* bg = sBattleGroundMgr.GetBattleGround(event.m_bgInstanceGuid, event.m_bgTypeId); - // battleground can be deleted already when we are removing queue info - // bg pointer can be nullptr! so use it carefully! - - uint32 queueSlot = plr->GetBattleGroundQueueIndex(event.m_bgQueueTypeId); - if (queueSlot < PLAYER_MAX_BATTLEGROUND_QUEUES) // player is in queue, or in Battleground - { - // check if player is in queue for this BG and if we are removing his invite event - BattleGroundQueue& bgQueue = sBattleGroundMgr.m_battleGroundQueues[event.m_bgQueueTypeId]; - if (bgQueue.IsPlayerInvited(event.m_playerGuid, event.m_bgInstanceGuid, event.m_removeTime)) - { - DEBUG_LOG("Battleground: removing player %u from bg queue for instance %u because of not pressing enter battle in time.", plr->GetGUIDLow(), event.m_bgInstanceGuid); - - plr->RemoveBattleGroundQueueId(event.m_bgQueueTypeId); - bgQueue.RemovePlayer(event.m_playerGuid, true); - - // update queues if battleground isn't ended - if (bg && bg->GetStatus() != STATUS_WAIT_LEAVE) - sBattleGroundMgr.ScheduleQueueUpdate(event.m_bgQueueTypeId, event.m_bgTypeId, bg->GetBracketId()); - - WorldPacket data; - sBattleGroundMgr.BuildBattleGroundStatusPacket(data, bg, queueSlot, STATUS_NONE, 0, 0); - plr->GetSession()->SendPacket(data); - } - } - }); - - // event will be deleted - return true; -} - -void BgQueueRemoveEvent::Abort(uint64 /*e_time*/) -{ - // do nothing -} - /*********************************************************/ /*** BATTLEGROUND MANAGER ***/ /*********************************************************/ -BattleGroundMgr::BattleGroundMgr() +BattleGroundMgr::BattleGroundMgr() : m_testing(false) { for (uint8 i = BATTLEGROUND_TYPE_NONE; i < MAX_BATTLEGROUND_TYPE_ID; ++i) m_battleGrounds[i].clear(); - m_testing = false; } BattleGroundMgr::~BattleGroundMgr() @@ -937,29 +67,9 @@ void BattleGroundMgr::DeleteAllBattleGrounds() @param diff */ -void BattleGroundMgr::Update(uint32 diff) +void BattleGroundMgr::Update(uint32 /*diff*/) { - // update scheduled queues - if (!m_queueUpdateScheduler.empty()) - { - std::vector scheduled; - { - // create mutex - // std::lock_guard guard(SchedulerLock); - // copy vector and clear the other - scheduled = std::vector(m_queueUpdateScheduler); - m_queueUpdateScheduler.clear(); - // release lock - } - - for (uint32 i : scheduled) - { - BattleGroundQueueTypeId bgQueueTypeId = BattleGroundQueueTypeId(i >> 16 & 255); - BattleGroundTypeId bgTypeId = BattleGroundTypeId((i >> 8) & 255); - BattleGroundBracketId bracketId = BattleGroundBracketId(i & 255); - m_battleGroundQueues[bgQueueTypeId].Update(bgTypeId, bracketId); - } - } + m_messager.Execute(this); } /** @@ -972,11 +82,11 @@ void BattleGroundMgr::Update(uint32 diff) @param time1 @param time2 */ -void BattleGroundMgr::BuildBattleGroundStatusPacket(WorldPacket& data, BattleGround* bg, uint8 queueSlot, uint8 statusId, uint32 time1, uint32 time2) const +void BattleGroundMgr::BuildBattleGroundStatusPacket(WorldPacket & data, bool bgExists, uint32 bgTypeId, uint32 bgClientInstanceId, uint32 mapId, uint8 queueSlot, uint8 statusId, uint32 time1, uint32 time2) { // we can be in 3 queues in same time... - if (statusId == 0 || !bg) + if (statusId == 0 || !bgExists) { data.Initialize(SMSG_BATTLEFIELD_STATUS, 4 * 2); data << uint32(queueSlot); // queue id (0...2) @@ -987,13 +97,13 @@ void BattleGroundMgr::BuildBattleGroundStatusPacket(WorldPacket& data, BattleGro data.Initialize(SMSG_BATTLEFIELD_STATUS, (4 + 8 + 4 + 1 + 4 + 4 + 4)); data << uint32(queueSlot); // queue id (0...2) - player can be in 3 queues in time // uint64 in client - data << uint32(bg->GetMapId()); // MapID + data << uint32(mapId); // MapID data << uint8(0); // Unknown - data << uint32(bg->GetClientInstanceId()); + data << uint32(bgClientInstanceId); data << uint32(statusId); // status switch (statusId) { - case STATUS_WAIT_QUEUE: // status_in_queue + case STATUS_WAIT_QUEUE: // status_in_queue data << uint32(time1); // average wait time, milliseconds data << uint32(time2); // time in queue, updated every minute!, milliseconds break; @@ -1016,7 +126,7 @@ void BattleGroundMgr::BuildBattleGroundStatusPacket(WorldPacket& data, BattleGro @param packet @param battleground */ -void BattleGroundMgr::BuildPvpLogDataPacket(WorldPacket& data, BattleGround* bg) const +void BattleGroundMgr::BuildPvpLogDataPacket(WorldPacket& data, BattleGround* bg) { data.Initialize(MSG_PVP_LOG_DATA, (1 + 4 + 40 * bg->GetPlayerScoresSize())); @@ -1083,7 +193,7 @@ void BattleGroundMgr::BuildPvpLogDataPacket(WorldPacket& data, BattleGround* bg) @param battleground type id @param battleground group join status */ -void BattleGroundMgr::BuildGroupJoinedBattlegroundPacket(WorldPacket& data, BattleGroundTypeId bgTypeId, BattleGroundGroupJoinStatus status) const +void BattleGroundMgr::BuildGroupJoinedBattlegroundPacket(WorldPacket& data, BattleGroundTypeId bgTypeId, BattleGroundGroupJoinStatus status) { data.Initialize(SMSG_GROUP_JOINED_BATTLEGROUND, 4); switch (status) @@ -1105,7 +215,7 @@ void BattleGroundMgr::BuildGroupJoinedBattlegroundPacket(WorldPacket& data, Batt @param field @param value */ -void BattleGroundMgr::BuildUpdateWorldStatePacket(WorldPacket& data, uint32 field, uint32 value) const +void BattleGroundMgr::BuildUpdateWorldStatePacket(WorldPacket& data, uint32 field, uint32 value) { data.Initialize(SMSG_UPDATE_WORLD_STATE, 4 + 4); data << uint32(field); @@ -1118,7 +228,7 @@ void BattleGroundMgr::BuildUpdateWorldStatePacket(WorldPacket& data, uint32 fiel @param packet @param sound id */ -void BattleGroundMgr::BuildPlaySoundPacket(WorldPacket& data, uint32 soundId) const +void BattleGroundMgr::BuildPlaySoundPacket(WorldPacket& data, uint32 soundId) { data.Initialize(SMSG_PLAY_SOUND, 4); data << uint32(soundId); @@ -1130,7 +240,7 @@ void BattleGroundMgr::BuildPlaySoundPacket(WorldPacket& data, uint32 soundId) co @param packet @param object guid */ -void BattleGroundMgr::BuildPlayerLeftBattleGroundPacket(WorldPacket& data, ObjectGuid guid) const +void BattleGroundMgr::BuildPlayerLeftBattleGroundPacket(WorldPacket& data, ObjectGuid guid) { data.Initialize(SMSG_BATTLEGROUND_PLAYER_LEFT, 8); data << ObjectGuid(guid); @@ -1142,7 +252,7 @@ void BattleGroundMgr::BuildPlayerLeftBattleGroundPacket(WorldPacket& data, Objec @param packet @param player */ -void BattleGroundMgr::BuildPlayerJoinedBattleGroundPacket(WorldPacket& data, Player* player) const +void BattleGroundMgr::BuildPlayerJoinedBattleGroundPacket(WorldPacket& data, Player* player) { data.Initialize(SMSG_BATTLEGROUND_PLAYER_JOINED, 8); data << player->GetObjectGuid(); @@ -1201,42 +311,21 @@ BattleGround* BattleGroundMgr::GetBattleGround(uint32 instanceId, BattleGroundTy @param battleground type id */ -BattleGround* BattleGroundMgr::GetBattleGroundTemplate(BattleGroundTypeId bgTypeId) +BattleGround* BattleGroundMgr::GetBattleGroundTemplate(BattleGroundTypeId bgTypeId) const { // map is sorted and we can be sure that lowest instance id has only BG template return m_battleGrounds[bgTypeId].empty() ? nullptr : m_battleGrounds[bgTypeId].begin()->second.get(); } /** - Function that returns client instance id from battleground type id and bracket id + Function that creates a new battleground that is actually used @param battleground type id - @param bracket id + @param bracket entry + @param arena type + @param isRated */ -uint32 BattleGroundMgr::CreateClientVisibleInstanceId(BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId) -{ - // we create here an instanceid, which is just for - // displaying this to the client and without any other use.. - // the client-instanceIds are unique for each battleground-type - // the instance-id just needs to be as low as possible, beginning with 1 - // the following works, because std::set is default ordered with "<" - // the optimalization would be to use as bitmask std::vector - but that would only make code unreadable - - uint32 lastId = 0; - ClientBattleGroundIdSet& ids = m_clientBattleGroundIds[bgTypeId][bracketId]; - for (ClientBattleGroundIdSet::const_iterator itr = ids.begin(); itr != ids.end();) - { - if ((++lastId) != *itr) // if there is a gap between the ids, we will break.. - break; - lastId = *itr; - } - ids.insert(lastId + 1); - - return lastId + 1; -} - -// create a new battleground that will really be used to play -BattleGround* BattleGroundMgr::CreateNewBattleGround(BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId) +BattleGround* BattleGroundMgr::CreateNewBattleGround(BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId, uint32 instanceId, uint32 clientInstanceId) { // get the template BG BattleGround* bgTemplate = GetBattleGroundTemplate(bgTypeId); @@ -1265,9 +354,9 @@ BattleGround* BattleGroundMgr::CreateNewBattleGround(BattleGroundTypeId bgTypeId } // will also set m_bgMap, instanceid - sMapMgr.CreateBgMap(bg->GetMapId(), bg); + sMapMgr.CreateBgMap(bg->GetMapId(), instanceId, bg); - bg->SetClientInstanceId(CreateClientVisibleInstanceId(bgTypeId, bracketId)); + bg->SetClientInstanceId(clientInstanceId); // reset the new bg (set status to status_wait_queue from status_none) bg->Reset(); @@ -1312,7 +401,7 @@ uint32 BattleGroundMgr::CreateBattleGround(BattleGroundTypeId bgTypeId, uint32 m } bg->SetMapId(mapId); - bg->SetTypeID(bgTypeId); + bg->SetTypeId(bgTypeId); bg->SetMinPlayersPerTeam(minPlayersPerTeam); bg->SetMaxPlayersPerTeam(maxPlayersPerTeam); bg->SetMinPlayers(minPlayersPerTeam * 2); @@ -1441,43 +530,6 @@ void BattleGroundMgr::CreateInitialBattleGrounds() sLog.outString(); } -/** - Method that builds battleground list data - - @param packet - @param battlemaster guid - @param player - @param battleground type id -*/ -void BattleGroundMgr::BuildBattleGroundListPacket(WorldPacket& data, ObjectGuid guid, Player* player, BattleGroundTypeId bgTypeId) const -{ - if (!player) - return; - - uint32 mapId = GetBattleGrounMapIdByTypeId(bgTypeId); - - data.Initialize(SMSG_BATTLEFIELD_LIST); - data << guid; // battlemaster guid - data << uint32(mapId); - data << uint8(0x00); // min level offset - - // battleground - indented for cross core compatibility - { - size_t count_pos = data.wpos(); - uint32 count = 0; - data << uint32(0); // number of bg instances - - uint32 bracket_id = player->GetBattleGroundBracketIdFromLevel(bgTypeId); - ClientBattleGroundIdSet const& ids = m_clientBattleGroundIds[bgTypeId][bracket_id]; - for (std::set::const_iterator itr = ids.begin(); itr != ids.end(); ++itr) - { - data << uint32(*itr); - ++count; - } - data.put(count_pos, count); - } -} - /** Method that sends player to battleground @@ -1557,33 +609,10 @@ void BattleGroundMgr::ToggleTesting() sWorld.SendWorldText(LANG_DEBUG_BG_ON); else sWorld.SendWorldText(LANG_DEBUG_BG_OFF); -} - -/** - Method that schedules queue update - - @param arena rating - @param arena type - @param battleground queue type id - @param battleground type id - @param bracket id -*/ -void BattleGroundMgr::ScheduleQueueUpdate(BattleGroundQueueTypeId bgQueueTypeId, BattleGroundTypeId bgTypeId, BattleGroundBracketId bracket_id) -{ - // std::lock_guard guard(SchedulerLock); - // we will use only 1 number created of bgTypeId and bracket_id - uint32 schedule_id = (bgQueueTypeId << 16) | (bgTypeId << 8) | bracket_id; - bool found = false; - for (unsigned long long i : m_queueUpdateScheduler) + sWorld.GetBGQueue().GetMessager().AddMessage([testing = m_testing](BattleGroundQueue* queue) { - if (i == schedule_id) - { - found = true; - break; - } - } - if (!found) - m_queueUpdateScheduler.push_back(schedule_id); + queue->SetTesting(testing); + }); } uint32 BattleGroundMgr::GetPrematureFinishTime() const @@ -1594,9 +623,9 @@ uint32 BattleGroundMgr::GetPrematureFinishTime() const /** Method that loads battlemaster entries from DB */ -void BattleGroundMgr::LoadBattleMastersEntry() +void BattleGroundMgr::LoadBattleMastersEntry(bool reload) { - m_battleMastersMap.clear(); // need for reload case + std::shared_ptr newBattleMastersMap = std::make_shared(); auto queryResult = WorldDatabase.Query("SELECT entry,bg_template FROM battlemaster_entry"); @@ -1628,10 +657,23 @@ void BattleGroundMgr::LoadBattleMastersEntry() continue; } - m_battleMastersMap[entry] = BattleGroundTypeId(bgTypeId); + (*newBattleMastersMap)[entry] = BattleGroundTypeId(bgTypeId); } while (queryResult->NextRow()); + m_battleMastersMap = newBattleMastersMap; + + if (reload) + { + sMapMgr.DoForAllMaps([battleMasters = newBattleMastersMap](Map* map) + { + map->GetMessager().AddMessage([battleMasters](Map* map) + { + map->GetMapDataContainer().SetBattleMastersMap(battleMasters); + }); + }); + } + sLog.outString(">> Loaded %u battlemaster entries", count); sLog.outString(); } @@ -1681,15 +723,15 @@ bool BattleGroundMgr::IsBgWeekend(BattleGroundTypeId bgTypeId) /** Method that loads battleground events used in battleground scripts */ -void BattleGroundMgr::LoadBattleEventIndexes() +void BattleGroundMgr::LoadBattleEventIndexes(bool reload) { BattleGroundEventIdx events; events.event1 = BG_EVENT_NONE; events.event2 = BG_EVENT_NONE; - m_gameObjectBattleEventIndexMap.clear(); // need for reload case - m_gameObjectBattleEventIndexMap[static_cast(-1)] = events; - m_creatureBattleEventIndexMap.clear(); // need for reload case - m_creatureBattleEventIndexMap[static_cast(-1)] = events; + std::shared_ptr newGameObjectIndexes = std::make_shared(); + (*newGameObjectIndexes)[static_cast(-1)] = events; + std::shared_ptr newCreatureIndexes = std::make_shared(); + (*newCreatureIndexes)[static_cast(-1)] = events; uint32 count = 0; @@ -1778,14 +820,64 @@ void BattleGroundMgr::LoadBattleEventIndexes() } if (gameobject) - m_gameObjectBattleEventIndexMap[dbTableGuidLow] = events; + (*newGameObjectIndexes)[dbTableGuidLow] = events; else - m_creatureBattleEventIndexMap[dbTableGuidLow] = events; + (*newCreatureIndexes)[dbTableGuidLow] = events; ++count; } while (queryResult->NextRow()); + m_gameObjectBattleEventIndexMap = newGameObjectIndexes; + m_creatureBattleEventIndexMap = newCreatureIndexes; + + if (reload) + { + sMapMgr.DoForAllMaps([gameobjects = newGameObjectIndexes, creatures = newCreatureIndexes](Map* map) + { + map->GetMessager().AddMessage([gameobjects, creatures](Map* map) + { + map->GetMapDataContainer().SetGameObjectEventIndexes(gameobjects); + map->GetMapDataContainer().SetCreatureEventIndexes(creatures); + }); + }); + } + sLog.outString(">> Loaded %u battleground eventindexes", count); sLog.outString(); } + +uint32 BattleGroundMgr::GetMinLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId) const +{ + if (bracket_id < 1) + return 0; + + if (bracket_id > BG_BRACKET_ID_LAST) + bracket_id = BG_BRACKET_ID_LAST; + + BattleGround* bg = GetBattleGroundTemplate(bgTypeId); + assert(bg); + return 10 * bracket_id + bg->GetMinLevel(); +} + +uint32 BattleGroundMgr::GetMaxLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId) const +{ + if (bracket_id >= BG_BRACKET_ID_LAST) + return 255; // hardcoded max level + + return GetMinLevelForBattleGroundBracketId(bracket_id, bgTypeId) + 10; +} + +BattleGroundBracketId BattleGroundMgr::GetBattleGroundBracketIdFromLevel(BattleGroundTypeId bgTypeId, uint32 playerLevel) const +{ + BattleGround* bg = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); + assert(bg); + if (playerLevel < bg->GetMinLevel()) + return BG_BRACKET_ID_FIRST; + + uint32 bracket_id = (playerLevel - bg->GetMinLevel()) / 10; + if (bracket_id > MAX_BATTLEGROUND_BRACKETS) + return BG_BRACKET_ID_LAST; + + return BattleGroundBracketId(bracket_id); +} diff --git a/src/game/BattleGround/BattleGroundMgr.h b/src/game/BattleGround/BattleGroundMgr.h index 8f8976cc3b..5ac8193441 100644 --- a/src/game/BattleGround/BattleGroundMgr.h +++ b/src/game/BattleGround/BattleGroundMgr.h @@ -23,169 +23,17 @@ #include "Policies/Singleton.h" #include "Util/UniqueTrackablePtr.h" #include "BattleGround.h" +#include "BattleGround/BattleGroundDefines.h" #include typedef std::map> BattleGroundSet; -// this container can't be deque, because deque doesn't like removing the last element - if you remove it, it invalidates next iterator and crash appears -typedef std::list BgFreeSlotQueueType; - typedef std::unordered_map BattleMastersMap; typedef std::unordered_map CreatureBattleEventIndexesMap; typedef std::unordered_map GameObjectBattleEventIndexesMap; -#define COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME 10 - -struct GroupQueueInfo; // type predefinition -struct PlayerQueueInfo // stores information for players in queue -{ - uint32 lastOnlineTime; // for tracking and removing offline players from queue after 5 minutes - GroupQueueInfo* groupInfo; // pointer to the associated groupqueueinfo -}; - -typedef std::map GroupQueueInfoPlayers; - -struct GroupQueueInfo // stores information about the group in queue (also used when joined as solo!) -{ - GroupQueueInfoPlayers players; // player queue info map - Team groupTeam; // Player team (ALLIANCE/HORDE) - BattleGroundTypeId bgTypeId; // battleground type id - uint32 joinTime; // time when group was added - uint32 removeInviteTime; // time when we will remove invite for players in group - uint32 isInvitedToBgInstanceGuid; // was invited to certain BG - uint32 desiredInstanceId; // queued for this instance specifically -}; - -enum BattleGroundQueueGroupTypes -{ - BG_QUEUE_PREMADE_ALLIANCE = 0, - BG_QUEUE_PREMADE_HORDE = 1, - BG_QUEUE_NORMAL_ALLIANCE = 2, - BG_QUEUE_NORMAL_HORDE = 3 -}; - -#define BG_QUEUE_GROUP_TYPES_COUNT 4 - -enum BattleGroundGroupJoinStatus -{ - BG_GROUP_JOIN_STATUS_DESERTERS = -2, - BG_GROUP_JOIN_STATUS_NOT_ELIGIBLE = -1, - BG_GROUP_JOIN_STATUS_SUCCESS = 0, -}; - class BattleGround; -class BattleGroundQueue -{ - public: - BattleGroundQueue(); - ~BattleGroundQueue(); - - void Update(BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketId*/); - - void FillPlayersToBg(BattleGround* /*bg*/, BattleGroundBracketId /*bracketId*/); - bool CheckPremadeMatch(BattleGroundBracketId /*bracketId*/, uint32 /*minPlayersPerTeam*/, uint32 /*maxPlayersPerTeam*/); - bool CheckNormalMatch(BattleGroundBracketId /*bracketId*/, uint32 /*minPlayers*/, uint32 /*maxPlayers*/); - GroupQueueInfo* AddGroup(Player* /*leader*/, Group* /*group*/, BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketEntry*/, bool /*isPremade*/, uint32 /*instanceId*/); - void RemovePlayer(ObjectGuid /*guid*/, bool /*decreaseInvitedCount*/); - bool IsPlayerInvited(ObjectGuid /*playerGuid*/, const uint32 /*bgInstanceGuid*/, const uint32 /*removeTime*/); - bool GetPlayerGroupInfoData(ObjectGuid /*guid*/, GroupQueueInfo* /*groupInfo*/); - void PlayerInvitedToBgUpdateAverageWaitTime(GroupQueueInfo* /*groupInfo*/, BattleGroundBracketId /*bracketId*/); - uint32 GetAverageQueueWaitTime(GroupQueueInfo* /*groupInfo*/, BattleGroundBracketId /*bracketId*/); - - private: - // mutex that should not allow changing private data, nor allowing to update Queue during private data change. - std::recursive_mutex m_lock; - - - typedef std::map QueuedPlayersMap; - QueuedPlayersMap m_queuedPlayers; - - // we need constant add to begin and constant remove / add from the end, therefore deque suits our problem well - typedef std::list GroupsQueueType; - - /* - This two dimensional array is used to store All queued groups - First dimension specifies the bgTypeId - Second dimension specifies the player's group types - - BG_QUEUE_PREMADE_ALLIANCE is used for premade alliance groups - BG_QUEUE_PREMADE_HORDE is used for premade horde groups - BG_QUEUE_NORMAL_ALLIANCE is used for normal (or small) alliance groups - BG_QUEUE_NORMAL_HORDE is used for normal (or small) horde groups - */ - GroupsQueueType m_queuedGroups[MAX_BATTLEGROUND_BRACKETS][BG_QUEUE_GROUP_TYPES_COUNT]; - - // class to select and invite groups to bg - class SelectionPool - { - public: - SelectionPool() : playerCount(0) {} - void Init(); - bool AddGroup(GroupQueueInfo* ginfo, uint32 desiredCount, uint32 bgInstanceId); - bool KickGroup(uint32 size); - uint32 GetPlayerCount() const {return playerCount;} - GroupsQueueType selectedGroups; - private: - uint32 playerCount; - }; - - // one selection pool for horde, other one for alliance - SelectionPool m_selectionPools[PVP_TEAM_COUNT]; - - bool InviteGroupToBg(GroupQueueInfo* /*groupInfo*/, BattleGround* /*bg*/, Team /*side*/); - - uint32 m_waitTimes[PVP_TEAM_COUNT][MAX_BATTLEGROUND_BRACKETS][COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME]; - uint32 m_waitTimeLastPlayer[PVP_TEAM_COUNT][MAX_BATTLEGROUND_BRACKETS]; - uint32 m_sumOfWaitTimes[PVP_TEAM_COUNT][MAX_BATTLEGROUND_BRACKETS]; -}; - -/* - This class is used to invite player to BG again, when minute lasts from his first invitation - it is capable to solve all possibilities -*/ -class BgQueueInviteEvent : public BasicEvent -{ - public: - BgQueueInviteEvent(ObjectGuid playerGuid, uint32 bgInstanceGuid, BattleGroundTypeId bgTypeId, uint32 removeTime) : - m_playerGuid(playerGuid), m_bgInstanceGuid(bgInstanceGuid), m_bgTypeId(bgTypeId), m_removeTime(removeTime) - { - }; - virtual ~BgQueueInviteEvent() {}; - - virtual bool Execute(uint64 e_time, uint32 p_time) override; - virtual void Abort(uint64 e_time) override; - - private: - ObjectGuid m_playerGuid; - uint32 m_bgInstanceGuid; - BattleGroundTypeId m_bgTypeId; - uint32 m_removeTime; -}; - -/* - This class is used to remove player from BG queue after 1 minute 20 seconds from first invitation - We must store removeInvite time in case player left queue and joined and is invited again - We must store bgQueueTypeId, because battleground can be deleted already, when player entered it -*/ -class BgQueueRemoveEvent : public BasicEvent -{ - public: - BgQueueRemoveEvent(ObjectGuid playerGuid, uint32 bgInstanceGuid, BattleGroundTypeId bgTypeId, BattleGroundQueueTypeId bgQueueTypeId, uint32 removeTime) - : m_playerGuid(playerGuid), m_bgInstanceGuid(bgInstanceGuid), m_removeTime(removeTime), m_bgTypeId(bgTypeId), m_bgQueueTypeId(bgQueueTypeId) - {} - - virtual ~BgQueueRemoveEvent() {} - - virtual bool Execute(uint64 e_time, uint32 p_time) override; - virtual void Abort(uint64 e_time) override; - - private: - ObjectGuid m_playerGuid; - uint32 m_bgInstanceGuid; - uint32 m_removeTime; - BattleGroundTypeId m_bgTypeId; - BattleGroundQueueTypeId m_bgQueueTypeId; -}; class BattleGroundMgr { @@ -196,78 +44,69 @@ class BattleGroundMgr void Update(uint32 /*diff*/); /* Packet Building */ - void BuildPlayerJoinedBattleGroundPacket(WorldPacket& /*data*/, Player* /*player*/) const; - void BuildPlayerLeftBattleGroundPacket(WorldPacket& /*data*/, ObjectGuid /*guid*/) const; - void BuildBattleGroundListPacket(WorldPacket& /*data*/, ObjectGuid /*guid*/, Player* /*player*/, BattleGroundTypeId /*bgTypeId*/) const; - void BuildGroupJoinedBattlegroundPacket(WorldPacket& /*data*/, BattleGroundTypeId /*bgTypeId*/, BattleGroundGroupJoinStatus /*status*/) const; - void BuildUpdateWorldStatePacket(WorldPacket& /*data*/, uint32 /*field*/, uint32 /*value*/) const; - void BuildPvpLogDataPacket(WorldPacket& /*data*/, BattleGround* /*bg*/) const; - void BuildBattleGroundStatusPacket(WorldPacket& /*data*/, BattleGround* /*bg*/, uint8 /*queueSlot*/, uint8 /*statusId*/, uint32 /*time1*/, uint32 /*time2*/) const; - void BuildPlaySoundPacket(WorldPacket& /*data*/, uint32 /*soundId*/) const; + static void BuildPlayerJoinedBattleGroundPacket(WorldPacket& /*data*/, Player* /*player*/); + static void BuildPlayerLeftBattleGroundPacket(WorldPacket& /*data*/, ObjectGuid /*guid*/); + static void BuildGroupJoinedBattlegroundPacket(WorldPacket& /*data*/, BattleGroundTypeId /*bgTypeId*/, BattleGroundGroupJoinStatus /*status*/); + static void BuildUpdateWorldStatePacket(WorldPacket& /*data*/, uint32 /*field*/, uint32 /*value*/); + static void BuildPvpLogDataPacket(WorldPacket& /*data*/, BattleGround* /*bg*/); + static void BuildBattleGroundStatusPacket(WorldPacket& data, bool bgExists, uint32 bgTypeId, uint32 bgClientInstanceId, uint32 mapId, uint8 queueSlot, uint8 statusId, uint32 time1, uint32 time2); + static void BuildPlaySoundPacket(WorldPacket& /*data*/, uint32 /*soundId*/); /* Battlegrounds */ BattleGround* GetBattleGroundThroughClientInstance(uint32 /*instanceId*/, BattleGroundTypeId /*bgTypeId*/); BattleGround* GetBattleGround(uint32 /*instanceId*/, BattleGroundTypeId /*bgTypeId*/); // there must be uint32 because MAX_BATTLEGROUND_TYPE_ID means unknown - BattleGround* GetBattleGroundTemplate(BattleGroundTypeId /*bgTypeId*/); - BattleGround* CreateNewBattleGround(BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketEntry*/); + BattleGround* GetBattleGroundTemplate(BattleGroundTypeId /*bgTypeId*/) const; + BattleGround* CreateNewBattleGround(BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketId*/, uint32 instanceId, uint32 clientInstanceId); uint32 CreateBattleGround(BattleGroundTypeId /*bgTypeId*/, uint32 /*minPlayersPerTeam*/, uint32 /*maxPlayersPerTeam*/, uint32 /*levelMin*/, uint32 /*levelMax*/, char const* /*battleGroundName*/, uint32 /*mapId*/, float /*team1StartLocX*/, float /*team1StartLocY*/, float /*team1StartLocZ*/, float /*team1StartLocO*/, float /*team2StartLocX*/, float /*team2StartLocY*/, float /*team2StartLocZ*/, float /*team2StartLocO*/, float /*startMaxDist*/, uint32 /*playerSkinReflootId*/); - void AddBattleGround(uint32 instanceId, BattleGroundTypeId bgTypeId, BattleGround* bg);; + void AddBattleGround(uint32 instanceId, BattleGroundTypeId bgTypeId, BattleGround* bg); void RemoveBattleGround(uint32 instanceId, BattleGroundTypeId bgTypeId) { m_battleGrounds[bgTypeId].erase(instanceId); } - uint32 CreateClientVisibleInstanceId(BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketId*/); - void DeleteClientVisibleInstanceId(BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId, uint32 clientInstanceId) - { - m_clientBattleGroundIds[bgTypeId][bracketId].erase(clientInstanceId); - } - void CreateInitialBattleGrounds(); void DeleteAllBattleGrounds(); void SendToBattleGround(Player* /*player*/, uint32 /*instanceId*/, BattleGroundTypeId /*bgTypeId*/); - /* Battleground queues */ - // these queues are instantiated when creating BattlegroundMrg - BattleGroundQueue m_battleGroundQueues[MAX_BATTLEGROUND_QUEUE_TYPES]; // public, because we need to access them in BG handler code - - BgFreeSlotQueueType BgFreeSlotQueue[MAX_BATTLEGROUND_TYPE_ID]; - - void ScheduleQueueUpdate(BattleGroundQueueTypeId /*bgQueueTypeId*/, BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketId*/); uint32 GetPrematureFinishTime() const; void ToggleTesting(); - void LoadBattleMastersEntry(); + void LoadBattleMastersEntry(bool reload); BattleGroundTypeId GetBattleMasterBG(uint32 entry) const { - BattleMastersMap::const_iterator itr = m_battleMastersMap.find(entry); - if (itr != m_battleMastersMap.end()) + BattleMastersMap::const_iterator itr = m_battleMastersMap->find(entry); + if (itr != m_battleMastersMap->end()) return itr->second; return BATTLEGROUND_TYPE_NONE; } + std::shared_ptr GetBattleMastersMap() const { return m_battleMastersMap; } - void LoadBattleEventIndexes(); + void LoadBattleEventIndexes(bool reload); const BattleGroundEventIdx GetCreatureEventIndex(uint32 dbGuid) const { - CreatureBattleEventIndexesMap::const_iterator itr = m_creatureBattleEventIndexMap.find(dbGuid); - if (itr != m_creatureBattleEventIndexMap.end()) + CreatureBattleEventIndexesMap::const_iterator itr = m_creatureBattleEventIndexMap->find(dbGuid); + if (itr != m_creatureBattleEventIndexMap->end()) return itr->second; - return m_creatureBattleEventIndexMap.find(static_cast(-1))->second; + return m_creatureBattleEventIndexMap->find(static_cast(-1))->second; } + std::shared_ptr GetCreatureEventIndexes() const { return m_creatureBattleEventIndexMap; } + const BattleGroundEventIdx GetGameObjectEventIndex(uint32 dbGuid) const { - GameObjectBattleEventIndexesMap::const_iterator itr = m_gameObjectBattleEventIndexMap.find(dbGuid); - if (itr != m_gameObjectBattleEventIndexMap.end()) + GameObjectBattleEventIndexesMap::const_iterator itr = m_gameObjectBattleEventIndexMap->find(dbGuid); + if (itr != m_gameObjectBattleEventIndexMap->end()) return itr->second; - return m_gameObjectBattleEventIndexMap.find(static_cast(-1))->second; + return m_gameObjectBattleEventIndexMap->find(static_cast(-1))->second; } + std::shared_ptr GetGameObjectEventIndexes() const { return m_gameObjectBattleEventIndexMap; } + bool IsTesting() const { return m_testing; } static BattleGroundQueueTypeId BgQueueTypeId(BattleGroundTypeId /*bgTypeId*/); @@ -278,19 +117,24 @@ class BattleGroundMgr static bool IsBgWeekend(BattleGroundTypeId /*bgTypeId*/); std::set const& GetUsedRefLootIds() const { return m_usedRefloot; } + + uint32 GetMinLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId) const; + uint32 GetMaxLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId) const; + BattleGroundBracketId GetBattleGroundBracketIdFromLevel(BattleGroundTypeId bgTypeId, uint32 playerLevel) const; + + Messager& GetMessager() { return m_messager; } private: std::mutex schedulerLock; - BattleMastersMap m_battleMastersMap; - CreatureBattleEventIndexesMap m_creatureBattleEventIndexMap; - GameObjectBattleEventIndexesMap m_gameObjectBattleEventIndexMap; + std::shared_ptr m_battleMastersMap; + std::shared_ptr m_creatureBattleEventIndexMap; + std::shared_ptr m_gameObjectBattleEventIndexMap; /* Battlegrounds */ BattleGroundSet m_battleGrounds[MAX_BATTLEGROUND_TYPE_ID]; - std::vector m_queueUpdateScheduler; - typedef std::set ClientBattleGroundIdSet; - ClientBattleGroundIdSet m_clientBattleGroundIds[MAX_BATTLEGROUND_TYPE_ID][MAX_BATTLEGROUND_BRACKETS]; // the instanceids just visible for the client - bool m_testing; + bool m_testing; std::set m_usedRefloot; + + Messager m_messager; }; #define sBattleGroundMgr MaNGOS::Singleton::Instance() diff --git a/src/game/BattleGround/BattleGroundQueue.cpp b/src/game/BattleGround/BattleGroundQueue.cpp new file mode 100644 index 0000000000..193b9006fa --- /dev/null +++ b/src/game/BattleGround/BattleGroundQueue.cpp @@ -0,0 +1,1262 @@ +/* + * This file is part of the CMaNGOS Project. See AUTHORS file for Copyright information + * + * 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 + * (at your option) 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, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include "BattleGround/BattleGroundQueue.h" +#include "Tools/Language.h" +#include "World/World.h" +#include "BattleGround/BattleGroundMgr.h" +#include "Globals/ObjectMgr.h" +#include "Entities/Player.h" +#include "Maps/MapManager.h" +#include "Server/WorldPacket.h" + + /*********************************************************/ + /*** BATTLEGROUND QUEUE SYSTEM ***/ + /*********************************************************/ + +BattleGroundQueueItem::BattleGroundQueueItem() +{ + for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) + { + for (uint8 j = 0; j < MAX_BATTLEGROUND_BRACKETS; ++j) + { + m_sumOfWaitTimes[i][j] = 0; + m_waitTimeLastPlayer[i][j] = 0; + + for (uint8 k = 0; k < COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME; ++k) + m_waitTimes[i][j][k] = 0; + } + } +} + +BattleGroundQueueItem::~BattleGroundQueueItem() +{ + m_queuedPlayers.clear(); + for (auto& m_queuedGroup : m_queuedGroups) + { + for (uint8 j = 0; j < BG_QUEUE_GROUP_TYPES_COUNT; ++j) + { + for (GroupsQueueType::iterator itr = m_queuedGroup[j].begin(); itr != m_queuedGroup[j].end(); ++itr) + delete (*itr); + + m_queuedGroup[j].clear(); + } + } +} + +/*********************************************************/ +/*** BATTLEGROUND QUEUE SELECTION POOLS ***/ +/*********************************************************/ + +// selection pool initialization, used to clean up from prev selection +void BattleGroundQueueItem::SelectionPool::Init() +{ + selectedGroups.clear(); + playerCount = 0; +} + +/** + Function that removes group infr from pool selection + - returns true when we need to try to add new group to selection pool + - returns false when selection pool is ok or when we kicked smaller group than we need to kick + - sometimes it can be called on empty selection pool + + @param size +*/ +bool BattleGroundQueueItem::SelectionPool::KickGroup(uint32 size) +{ + // find maxgroup or LAST group with size == size and kick it + bool found = false; + GroupsQueueType::iterator groupToKick = selectedGroups.begin(); + + for (GroupsQueueType::iterator itr = groupToKick; itr != selectedGroups.end(); ++itr) + { + if (abs((int32)((*itr)->players.size() - size)) <= 1) + { + groupToKick = itr; + found = true; + } + else if (!found && (*itr)->players.size() >= (*groupToKick)->players.size()) + groupToKick = itr; + } + + // if pool is empty, do nothing + if (GetPlayerCount()) + { + // update player count + GroupQueueInfo* queueInfo = (*groupToKick); + selectedGroups.erase(groupToKick); + playerCount -= queueInfo->players.size(); + + // return false if we kicked smaller group or there are enough players in selection pool + if (queueInfo->players.size() <= size + 1) + return false; + } + return true; +} + +/** + Function that adds group to selection pool + - returns true if we can invite more players, or when we added group to selection pool + - returns false when selection pool is full + + @param group queue info + @param desired count +*/ +bool BattleGroundQueueItem::SelectionPool::AddGroup(GroupQueueInfo* queueInfo, uint32 desiredCount, uint32 bgInstanceId) +{ + // if group is larger than desired count - don't allow to add it to pool + if (!queueInfo->isInvitedToBgInstanceGuid && + (!queueInfo->desiredInstanceId || queueInfo->desiredInstanceId == bgInstanceId) && + (desiredCount >= playerCount + queueInfo->players.size())) + { + selectedGroups.push_back(queueInfo); + // increase selected players count + playerCount += queueInfo->players.size(); + + return true; + } + + return playerCount < desiredCount; +} + +/*********************************************************/ +/*** BATTLEGROUND QUEUES ***/ +/*********************************************************/ + +/** + Function that adds group or player (grp == nullptr) to battleground queue with the given leader and specifications + + @param leader player + @param group + @param battleground type id + @param bracket entry + @param arena type + @param isRated + @param isPremade + @param arena rating + @param arena team id +*/ +GroupQueueInfo* BattleGroundQueueItem::AddGroup(ObjectGuid leader, AddGroupToQueueInfo const& groupInfo, BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId, bool isPremade, uint32 instanceId) +{ + // create new ginfo + GroupQueueInfo* queueInfo = new GroupQueueInfo; + queueInfo->bgTypeId = bgTypeId; + queueInfo->bgBracketId = bracketId; + queueInfo->isInvitedToBgInstanceGuid = 0; + queueInfo->mapId = groupInfo.mapId; + queueInfo->clientInstanceId = groupInfo.clientInstanceId; + queueInfo->joinTime = WorldTimer::getMSTime(); + queueInfo->removeInviteTime = 0; + queueInfo->groupTeam = groupInfo.team; + queueInfo->desiredInstanceId = instanceId; + + queueInfo->players.clear(); + + // compute index (if group is premade or joined a rated match) to queues + uint32 index = 0; + if (!isPremade) + index += PVP_TEAM_COUNT; // BG_QUEUE_PREMADE_* -> BG_QUEUE_NORMAL_* + + if (queueInfo->groupTeam == HORDE) + ++index; // BG_QUEUE_*_ALLIANCE -> BG_QUEUE_*_HORDE + + DEBUG_LOG("Adding Group to BattleGroundQueueItem bgTypeId : %u, bracket_id : %u, index : %u", bgTypeId, bracketId, index); + + uint32 lastOnlineTime = WorldTimer::getMSTime(); + + // add players from group to ginfo + { + if (!groupInfo.members.empty()) + { + for (ObjectGuid member : groupInfo.members) + { + PlayerQueueInfo& playerInfo = m_queuedPlayers[member]; + playerInfo.lastOnlineTime = lastOnlineTime; + playerInfo.groupInfo = queueInfo; + // add the pinfo to ginfo's list + queueInfo->players[member] = &playerInfo; + } + } + else + { + PlayerQueueInfo& playerInfo = m_queuedPlayers[leader]; + playerInfo.lastOnlineTime = lastOnlineTime; + playerInfo.groupInfo = queueInfo; + queueInfo->players[leader] = &playerInfo; + } + + // add GroupInfo to m_QueuedGroups + m_queuedGroups[bracketId][index].push_back(queueInfo); + + // announce to world, this code needs mutex + if (!isPremade && sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_QUEUE_ANNOUNCER_JOIN)) + { + if (BattleGround* bg = sBattleGroundMgr.GetBattleGroundTemplate(queueInfo->bgTypeId)) + { + char const* bgName = bg->GetName(); + uint32 minPlayers = bg->GetMinPlayersPerTeam(); + uint32 qHorde = 0; + uint32 qAlliance = 0; + uint32 q_min_level = sBattleGroundMgr.GetMinLevelForBattleGroundBracketId(bracketId, bgTypeId); + uint32 qMaxLevel = sBattleGroundMgr.GetMaxLevelForBattleGroundBracketId(bracketId, bgTypeId); + GroupsQueueType::const_iterator itr; + for (itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].begin(); itr != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].end(); ++itr) + if (!(*itr)->isInvitedToBgInstanceGuid) + qAlliance += (*itr)->players.size(); + + for (itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].begin(); itr != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].end(); ++itr) + if (!(*itr)->isInvitedToBgInstanceGuid) + qHorde += (*itr)->players.size(); + + sWorld.GetMessager().AddMessage([playerGuid = leader, bgName, q_min_level, qMaxLevel, qAlliance, minPlayers, qHorde](World* world) + { + // Show queue status to player only (when joining queue) + if (sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_QUEUE_ANNOUNCER_JOIN) == 1) + { + if (Player* plr = sObjectMgr.GetPlayer(playerGuid)) + ChatHandler(plr).PSendSysMessage(LANG_BG_QUEUE_ANNOUNCE_SELF, bgName, q_min_level, qMaxLevel, + qAlliance, (minPlayers > qAlliance) ? minPlayers - qAlliance : (uint32)0, qHorde, (minPlayers > qHorde) ? minPlayers - qHorde : (uint32)0); + } + // System message + else + { + sWorld.SendWorldText(LANG_BG_QUEUE_ANNOUNCE_WORLD, bgName, q_min_level, qMaxLevel, + qAlliance, (minPlayers > qAlliance) ? minPlayers - qAlliance : (uint32)0, qHorde, (minPlayers > qHorde) ? minPlayers - qHorde : (uint32)0); + } + }); + } + } + // release mutex + } + + return queueInfo; +} + +/** + Method that updates average update wait time + + @param group queue info + @param bracket id +*/ +void BattleGroundQueueItem::PlayerInvitedToBgUpdateAverageWaitTime(GroupQueueInfo* queueInfo, BattleGroundBracketId bracketId) +{ + uint32 timeInQueue = WorldTimer::getMSTimeDiff(queueInfo->joinTime, WorldTimer::getMSTime()); + uint8 teamIndex = TEAM_INDEX_ALLIANCE; // default set to BG_TEAM_ALLIANCE - or non rated arenas! + + if (queueInfo->groupTeam == HORDE) + teamIndex = TEAM_INDEX_HORDE; + + // store pointer to arrayindex of player that was added first + uint32* lastPlayerAddedPointer = &(m_waitTimeLastPlayer[teamIndex][bracketId]); + + // remove his time from sum + m_sumOfWaitTimes[teamIndex][bracketId] -= m_waitTimes[teamIndex][bracketId][(*lastPlayerAddedPointer)]; + + // set average time to new + m_waitTimes[teamIndex][bracketId][(*lastPlayerAddedPointer)] = timeInQueue; + + // add new time to sum + m_sumOfWaitTimes[teamIndex][bracketId] += timeInQueue; + + // set index of last player added to next one + (*lastPlayerAddedPointer)++; + (*lastPlayerAddedPointer) %= COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME; +} + +/** + Function that returns averate queue wait time + + @param group queue info + @param bracket id +*/ +uint32 BattleGroundQueueItem::GetAverageQueueWaitTime(GroupQueueInfo* queueInfo, BattleGroundBracketId bracketId) +{ + uint8 teamIndex = TEAM_INDEX_ALLIANCE; // default set to BG_TEAM_ALLIANCE - or non rated arenas! + if (queueInfo->groupTeam == HORDE) + teamIndex = TEAM_INDEX_HORDE; + + // check if there is enought values(we always add values > 0) + if (m_waitTimes[teamIndex][bracketId][COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME - 1]) + return (m_sumOfWaitTimes[teamIndex][bracketId] / COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME); + + // if there aren't enough values return 0 - not available + return 0; +} + +/** + Method that removes player from queue and from group info, if group info is empty then remove it too + + @param guid + @param decrease invite count +*/ +void BattleGroundQueueItem::RemovePlayer(BattleGroundQueue& queue, ObjectGuid guid, bool decreaseInvitedCount) +{ + int32 bracketId = -1; // signed for proper for-loop finish + + // remove player from map, if he's there + QueuedPlayersMap::iterator itr = m_queuedPlayers.find(guid); + if (itr == m_queuedPlayers.end()) + { + sLog.outError("BattleGroundQueueItem: couldn't find for remove: %s", guid.GetString().c_str()); + return; + } + + GroupQueueInfo* group = itr->second.groupInfo; + GroupsQueueType::iterator group_itr; + // mostly people with the highest levels are in battlegrounds, thats why + // we count from MAX_BATTLEGROUND_QUEUES - 1 to 0 + // variable index removes useless searching in other team's queue + uint32 index = GetTeamIndexByTeamId(group->groupTeam); + + for (int8 bracketIdTmp = MAX_BATTLEGROUND_BRACKETS - 1; bracketIdTmp >= 0 && bracketId == -1; --bracketIdTmp) + { + // we must check premade and normal team's queue - because when players from premade are joining bg, + // they leave groupinfo so we can't use its players size to find out index + for (uint8 j = index; j < BG_QUEUE_GROUP_TYPES_COUNT; j += BG_QUEUE_NORMAL_ALLIANCE) + { + for (GroupsQueueType::iterator group_itr_tmp = m_queuedGroups[bracketIdTmp][j].begin(); group_itr_tmp != m_queuedGroups[bracketIdTmp][j].end(); ++group_itr_tmp) + { + if ((*group_itr_tmp) == group) + { + bracketId = bracketIdTmp; + group_itr = group_itr_tmp; + // we must store index to be able to erase iterator + index = j; + break; + } + } + } + } + + // player can't be in queue without group, but just in case + if (bracketId == -1) + { + sLog.outError("BattleGroundQueueItem: ERROR Cannot find groupinfo for %s", guid.GetString().c_str()); + return; + } + DEBUG_LOG("BattleGroundQueueItem: Removing %s, from bracket_id %u", guid.GetString().c_str(), (uint32)bracketId); + + // ALL variables are correctly set + // We can ignore leveling up in queue - it should not cause crash + // remove player from group + // if only one player there, remove group + + // remove player queue info from group queue info + GroupQueueInfoPlayers::iterator pitr = group->players.find(guid); + if (pitr != group->players.end()) + group->players.erase(pitr); + + // if invited to bg, and should decrease invited count, then do it + if (decreaseInvitedCount && group->isInvitedToBgInstanceGuid) + { + if (BattleGroundInQueueInfo* bgInstance = queue.GetFreeSlotInstance(group->bgTypeId, group->isInvitedToBgInstanceGuid)) + bgInstance->DecreaseInvitedCount(group->groupTeam); + } + + // remove player queue info + m_queuedPlayers.erase(itr); + + // remove group queue info if needed + if (group->players.empty()) + { + m_queuedGroups[bracketId][index].erase(group_itr); + delete group; + } +} + +/** + Function that returns true when player is in queue and is invited to bgInstanceGuid + + @param player guid + @param battleground instance guid + @param remove time +*/ +bool BattleGroundQueueItem::IsPlayerInvited(ObjectGuid playerGuid, const uint32 bgInstanceGuid, const uint32 removeTime) +{ + QueuedPlayersMap::const_iterator qItr = m_queuedPlayers.find(playerGuid); + return (qItr != m_queuedPlayers.end() + && qItr->second.groupInfo->isInvitedToBgInstanceGuid == bgInstanceGuid + && qItr->second.groupInfo->removeInviteTime == removeTime); +} + +/** + Function that returns player group info data + - returns true when the player is found in queue + + @param player guid + @param group queue info +*/ +bool BattleGroundQueueItem::GetPlayerGroupInfoData(ObjectGuid guid, GroupQueueInfo* queueInfo) +{ + // std::lock_guard g(m_Lock); + QueuedPlayersMap::const_iterator qItr = m_queuedPlayers.find(guid); + if (qItr == m_queuedPlayers.end()) + return false; + + *queueInfo = *(qItr->second.groupInfo); + return true; +} + +/** + Function that invites group to battleground + + @param group queue info + @param battleground + @param team +*/ +bool BattleGroundQueueItem::InviteGroupToBg(GroupQueueInfo* groupInfo, BattleGroundInQueueInfo& queueInfo, Team team) +{ + // set side if needed + if (team == ALLIANCE || team == HORDE) + groupInfo->groupTeam = team; + + if (!groupInfo->isInvitedToBgInstanceGuid) + { + // not yet invited + // set invitation + groupInfo->isInvitedToBgInstanceGuid = queueInfo.GetInstanceId(); + groupInfo->mapId = queueInfo.GetMapId(); + groupInfo->clientInstanceId = queueInfo.GetClientInstanceId(); + BattleGroundTypeId bgTypeId = queueInfo.GetTypeId(); + BattleGroundQueueTypeId bgQueueTypeId = BattleGroundMgr::BgQueueTypeId(bgTypeId); + BattleGroundBracketId bracket_id = queueInfo.GetBracketId(); + + groupInfo->removeInviteTime = WorldTimer::getMSTime() + INVITE_ACCEPT_WAIT_TIME; + + // loop through the players + for (auto itr = groupInfo->players.begin(); itr != groupInfo->players.end(); ++itr) + { + sWorld.GetMessager().AddMessage([playerGuid = itr->first, bgQueueTypeId, bgTypeId, isInvited = groupInfo->isInvitedToBgInstanceGuid, clientInstanceId = queueInfo.GetClientInstanceId(), mapId = queueInfo.GetMapId(), removeInviteTime = groupInfo->removeInviteTime, instanceId = queueInfo.GetInstanceId(), isBg = queueInfo.IsBattleGround()](World* world) + { + Player* plr = sObjectMgr.GetPlayer(playerGuid); + // if offline, skip him, can happen due to asynchronicity now + if (!plr) + return; + + plr->SetInviteForBattleGroundQueueType(bgQueueTypeId, isInvited); + + // create remind invite events + BgQueueInviteEvent* inviteEvent = new BgQueueInviteEvent(plr->GetObjectGuid(), isInvited, bgTypeId, removeInviteTime); + plr->m_events.AddEvent(inviteEvent, plr->m_events.CalculateTime(INVITATION_REMIND_TIME)); + + // create automatic remove events + BgQueueRemoveEvent* removeEvent = new BgQueueRemoveEvent(plr->GetObjectGuid(), isInvited, bgTypeId, bgQueueTypeId, removeInviteTime); + plr->m_events.AddEvent(removeEvent, plr->m_events.CalculateTime(INVITE_ACCEPT_WAIT_TIME)); + + WorldPacket data; + + uint32 queueSlot = plr->GetBattleGroundQueueIndex(bgQueueTypeId); + + DEBUG_LOG("Battleground: invited %s to BG instance %u queueindex %u bgtype %u, I can't help it if they don't press the enter battle button.", plr->GetGuidStr().c_str(), instanceId, queueSlot, bgTypeId); + + // send status packet + sBattleGroundMgr.BuildBattleGroundStatusPacket(data, isBg, bgTypeId, clientInstanceId, mapId, queueSlot, STATUS_WAIT_JOIN, INVITE_ACCEPT_WAIT_TIME, 0); + + plr->GetSession()->SendPacket(data); + }); + + PlayerInvitedToBgUpdateAverageWaitTime(groupInfo, bracket_id); + + // set invited player counters + queueInfo.IncreaseInvitedCount(groupInfo->groupTeam); + // if issues arise due to async state, need to add pending and confirmation + } + return true; + } + + return false; +} + +/** + Method that invites players to an already running battleground + - invitation is based on config file + - large groups are disadvantageous, because they will be kicked first if invitation type = 1 + + @param battleground + @param bracket id +*/ +void BattleGroundQueueItem::FillPlayersToBg(BattleGroundInQueueInfo& queueInfo, BattleGroundBracketId bracketId) +{ + int32 hordeFree = queueInfo.GetFreeSlotsForTeam(HORDE); + int32 aliFree = queueInfo.GetFreeSlotsForTeam(ALLIANCE); + + // iterator for iterating through bg queue + GroupsQueueType::const_iterator Ali_itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].begin(); + // count of groups in queue - used to stop cycles + uint32 aliCount = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].size(); + // index to queue which group is current + uint32 aliIndex = 0; + for (; aliIndex < aliCount && m_selectionPools[TEAM_INDEX_ALLIANCE].AddGroup((*Ali_itr), aliFree, queueInfo.GetClientInstanceId()); ++aliIndex) + ++Ali_itr; + + // the same thing for horde + GroupsQueueType::const_iterator Horde_itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].begin(); + uint32 hordeCount = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].size(); + uint32 hordeIndex = 0; + for (; hordeIndex < hordeCount && m_selectionPools[TEAM_INDEX_HORDE].AddGroup((*Horde_itr), hordeFree, queueInfo.GetClientInstanceId()); ++hordeIndex) + ++Horde_itr; + + // if ofc like BG queue invitation is set in config, then we are happy + if (sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_INVITATION_TYPE) == 0) + return; + + /* + if we reached this code, then we have to solve NP - complete problem called Subset sum problem + So one solution is to check all possible invitation subgroups, or we can use these conditions: + 1. Last time when BattleGroundQueueItem::Update was executed we invited all possible players - so there is only small possibility + that we will invite now whole queue, because only 1 change has been made to queues from the last BattleGroundQueueItem::Update call + 2. Other thing we should consider is group order in queue + */ + + // At first we need to compare free space in bg and our selection pool + int32 diffAli = aliFree - int32(m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()); + int32 diffHorde = hordeFree - int32(m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()); + while (abs(diffAli - diffHorde) > 1 && (m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() > 0 || m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() > 0)) + { + // each cycle execution we need to kick at least 1 group + if (diffAli < diffHorde) + { + // kick alliance group, add to pool new group if needed + if (m_selectionPools[TEAM_INDEX_ALLIANCE].KickGroup(diffHorde - diffAli)) + { + for (; aliIndex < aliCount && m_selectionPools[TEAM_INDEX_ALLIANCE].AddGroup((*Ali_itr), (aliFree >= diffHorde) ? aliFree - diffHorde : 0, queueInfo.GetClientInstanceId()); ++aliIndex) + ++Ali_itr; + } + // if ali selection is already empty, then kick horde group, but if there are less horde than ali in bg - break; + if (!m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()) + { + if (aliFree <= diffHorde + 1) + break; + m_selectionPools[TEAM_INDEX_HORDE].KickGroup(diffHorde - diffAli); + } + } + else + { + // kick horde group, add to pool new group if needed + if (m_selectionPools[TEAM_INDEX_HORDE].KickGroup(diffAli - diffHorde)) + { + for (; hordeIndex < hordeCount && m_selectionPools[TEAM_INDEX_HORDE].AddGroup((*Horde_itr), (hordeFree >= diffAli) ? hordeFree - diffAli : 0, queueInfo.GetClientInstanceId()); ++hordeIndex) + ++Horde_itr; + } + if (!m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()) + { + if (hordeFree <= diffAli + 1) + break; + m_selectionPools[TEAM_INDEX_ALLIANCE].KickGroup(diffAli - diffHorde); + } + } + + // count diffs after small update + diffAli = aliFree - int32(m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()); + diffHorde = hordeFree - int32(m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()); + } +} + +/** + Method that checks if (premade vs premade) battlegrouns is possible + - then after 30 mins (default) in queue it moves premade group to normal queue + - it tries to invite as much players as it can - to maxPlayersPerTeam, because premade groups have more than minPlayersPerTeam players + + @param bracket id + @param min players per team + @param max players per team +*/ +bool BattleGroundQueueItem::CheckPremadeMatch(BattleGroundBracketId bracketId, uint32 minPlayersPerTeam, uint32 maxPlayersPerTeam) +{ + // check match + if (!m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].empty() && !m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].empty()) + { + // start premade match + // if groups aren't invited + GroupsQueueType::const_iterator ali_group, horde_group; + for (ali_group = m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].begin(); ali_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].end(); ++ali_group) + if (!(*ali_group)->isInvitedToBgInstanceGuid) + break; + + for (horde_group = m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].begin(); horde_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].end(); ++horde_group) + if (!(*horde_group)->isInvitedToBgInstanceGuid) + break; + + if (ali_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].end() && horde_group != m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].end()) + { + m_selectionPools[TEAM_INDEX_ALLIANCE].AddGroup((*ali_group), maxPlayersPerTeam, 0); + m_selectionPools[TEAM_INDEX_HORDE].AddGroup((*horde_group), maxPlayersPerTeam, 0); + + // add groups/players from normal queue to size of bigger group + uint32 maxPlayers = std::max(m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount(), m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount()); + for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) + { + for (GroupsQueueType::const_iterator itr = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].begin(); itr != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].end(); ++itr) + { + // if itr can join BG and player count is less that maxPlayers, then add group to selectionpool + if (!(*itr)->isInvitedToBgInstanceGuid && !m_selectionPools[i].AddGroup((*itr), maxPlayers, 0)) + break; + } + } + + // premade selection pools are set + return true; + } + } + // now check if we can move group from Premade queue to normal queue (timer has expired) or group size lowered!! + // this could be 2 cycles but i'm checking only first team in queue - it can cause problem - + // if first is invited to BG and seconds timer expired, but we can ignore it, because players have only 80 seconds to click to enter bg + // and when they click or after 80 seconds the queue info is removed from queue + uint32 time_before = WorldTimer::getMSTime() - sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_PREMADE_GROUP_WAIT_FOR_MATCH); + for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) + { + if (!m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE + i].empty()) + { + GroupsQueueType::iterator itr = m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE + i].begin(); + if (!(*itr)->isInvitedToBgInstanceGuid && ((*itr)->joinTime < time_before || (*itr)->players.size() < minPlayersPerTeam)) + { + // we must insert group to normal queue and erase pointer from premade queue + m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].push_front((*itr)); + m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE + i].erase(itr); + } + } + } + + // selection pools are not set + return false; +} + +/** + Method that tries to create battleground or arena with minPlayersPerTeam against maxPlayersPerTeam + + @param battleground + @param bracket id + @param min players + @param max players +*/ +bool BattleGroundQueueItem::CheckNormalMatch(BattleGroundQueue& queue, BattleGround* bgTemplate, BattleGroundBracketId bracketId, uint32 minPlayers, uint32 maxPlayers) +{ + GroupsQueueType::const_iterator itr_team[PVP_TEAM_COUNT]; + for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) + { + itr_team[i] = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].begin(); + for (; itr_team[i] != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + i].end(); ++(itr_team[i])) + { + if (!(*(itr_team[i]))->isInvitedToBgInstanceGuid) + { + m_selectionPools[i].AddGroup(*(itr_team[i]), maxPlayers, 0); + if (m_selectionPools[i].GetPlayerCount() >= minPlayers) + break; + } + } + } + + // try to invite same number of players - this cycle may cause longer wait time even if there are enough players in queue, but we want ballanced bg + uint32 j = TEAM_INDEX_ALLIANCE; + if (m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() < m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount()) + j = TEAM_INDEX_HORDE; + + if (sWorld.getConfig(CONFIG_UINT32_BATTLEGROUND_INVITATION_TYPE) != 0 + && m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() >= minPlayers && m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() >= minPlayers) + { + // we will try to invite more groups to team with less players indexed by j + ++(itr_team[j]); // this will not cause a crash, because for cycle above reached break; + for (; itr_team[j] != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + j].end(); ++(itr_team[j])) + { + if (!(*(itr_team[j]))->isInvitedToBgInstanceGuid) + if (!m_selectionPools[j].AddGroup(*(itr_team[j]), m_selectionPools[(j + 1) % PVP_TEAM_COUNT].GetPlayerCount(), 0)) + break; + } + // do not allow to start bg with more than 2 players more on 1 faction + if (abs((int32)(m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() - m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount())) > 2) + return false; + } + + // allow 1v0 if debug bg + if (queue.IsTesting() && (m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() || m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount())) + return true; + + // return true if there are enough players in selection pools - enable to work .debug bg command correctly + return m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() >= minPlayers && m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() >= minPlayers; +} + +/** + Method that will check if we can invite players to same faction skirmish match + + @param bracket id + @param min players +*/ +bool BattleGroundQueueItem::CheckSkirmishForSameFaction(BattleGroundBracketId bracketId, uint32 minPlayersPerTeam) +{ + if (m_selectionPools[TEAM_INDEX_ALLIANCE].GetPlayerCount() < minPlayersPerTeam && m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() < minPlayersPerTeam) + return false; + + PvpTeamIndex teamIdx = TEAM_INDEX_ALLIANCE; + PvpTeamIndex otherTeamIdx = TEAM_INDEX_HORDE; + Team otherTeamId = HORDE; + + if (m_selectionPools[TEAM_INDEX_HORDE].GetPlayerCount() == minPlayersPerTeam) + { + teamIdx = TEAM_INDEX_HORDE; + otherTeamIdx = TEAM_INDEX_ALLIANCE; + otherTeamId = ALLIANCE; + } + + // clear other team's selection + m_selectionPools[otherTeamIdx].Init(); + // store last ginfo pointer + GroupQueueInfo* ginfo = m_selectionPools[teamIdx].selectedGroups.back(); + // set itr_team to group that was added to selection pool latest + GroupsQueueType::iterator itr_team = m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + teamIdx].begin(); + for (; itr_team != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + teamIdx].end(); ++itr_team) + if (ginfo == *itr_team) + break; + + if (itr_team == m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + teamIdx].end()) + return false; + + GroupsQueueType::iterator itr_team2 = itr_team; + ++itr_team2; + // invite players to other selection pool + for (; itr_team2 != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + teamIdx].end(); ++itr_team2) + { + // if selection pool is full then break; + if (!(*itr_team2)->isInvitedToBgInstanceGuid && !m_selectionPools[otherTeamIdx].AddGroup(*itr_team2, minPlayersPerTeam, 0)) + break; + } + + if (m_selectionPools[otherTeamIdx].GetPlayerCount() != minPlayersPerTeam) + return false; + + // here we have correct 2 selections and we need to change one teams team and move selection pool teams to other team's queue + for (GroupsQueueType::iterator itr = m_selectionPools[otherTeamIdx].selectedGroups.begin(); itr != m_selectionPools[otherTeamIdx].selectedGroups.end(); ++itr) + { + // set correct team + (*itr)->groupTeam = otherTeamId; + + // add team to other queue + m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + otherTeamIdx].push_front(*itr); + + // remove team from old queue + GroupsQueueType::iterator itr2 = itr_team; + ++itr2; + for (; itr2 != m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + teamIdx].end(); ++itr2) + { + if (*itr2 == *itr) + { + m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE + teamIdx].erase(itr2); + break; + } + } + } + return true; +} + +/** + Method that is called when group is inserted, or player / group is removed from BG Queue - there is only one player's status changed, so we don't use while(true) cycles to invite whole queue + - it must be called after fully adding the members of a group to ensure group joining + - should be called from BattleGround::RemovePlayer function in some cases + + @param bg type id + @param bracket id + @param arena type + @param isRated + @param arenaRating +*/ +void BattleGroundQueueItem::Update(BattleGroundQueue& queue, BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId) +{ + // if no players in queue - do nothing + if (m_queuedGroups[bracketId][BG_QUEUE_PREMADE_ALLIANCE].empty() && + m_queuedGroups[bracketId][BG_QUEUE_PREMADE_HORDE].empty() && + m_queuedGroups[bracketId][BG_QUEUE_NORMAL_ALLIANCE].empty() && + m_queuedGroups[bracketId][BG_QUEUE_NORMAL_HORDE].empty()) + return; + + // battleground with free slot for player should be always in the beggining of the queue + // maybe it would be better to create bgfreeslotqueue for each bracket_id + BgFreeSlotQueueType::iterator next; + auto queueItems = queue.GetFreeSlotQueueItem(bgTypeId); + for (BgFreeSlotQueueType::iterator itr = queueItems.begin(); itr != queueItems.end(); itr = next) + { + BattleGroundInQueueInfo& queueInfo = *itr; + next = itr; + ++next; + // DO NOT allow queue manager to invite new player to arena + if (queueInfo.IsBattleGround() && queueInfo.GetTypeId() == bgTypeId && queueInfo.GetBracketId() == bracketId && + queueInfo.GetStatus() > STATUS_WAIT_QUEUE && queueInfo.GetStatus() < STATUS_WAIT_LEAVE) + { + // and iterator is invalid + + // clear selection pools + m_selectionPools[TEAM_INDEX_ALLIANCE].Init(); + m_selectionPools[TEAM_INDEX_HORDE].Init(); + + // call a function that does the job for us + FillPlayersToBg(queueInfo, bracketId); + + // now everything is set, invite players + for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_ALLIANCE].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_ALLIANCE].selectedGroups.end(); ++citr) + InviteGroupToBg((*citr), queueInfo, (*citr)->groupTeam); + for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_HORDE].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_HORDE].selectedGroups.end(); ++citr) + InviteGroupToBg((*citr), queueInfo, (*citr)->groupTeam); + + if (!queueInfo.HasFreeSlots()) + { + // remove BG from BGFreeSlotQueue + queueItems.erase(itr); + sWorld.GetMessager().AddMessage([instanceId = queueInfo.instanceId, typeId = queueInfo.bgTypeId](World* world) + { + if (BattleGround* bg = sBattleGroundMgr.GetBattleGround(instanceId, typeId)) + { + bg->RemovedFromBgFreeSlotQueue(false); + } + }); + } + } + } + + // finished iterating through the bgs with free slots, maybe we need to create a new bg + + BattleGround* bgTemplate = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); + if (!bgTemplate) + { + sLog.outError("Battleground: Update: bg template not found for %u", bgTypeId); + return; + } + + // get the min. players per team, properly for larger arenas as well. (must have full teams for arena matches!) + uint32 minPlayersPerTeam = bgTemplate->GetMinPlayersPerTeam(); + uint32 maxPlayersPerTeam = bgTemplate->GetMaxPlayersPerTeam(); + + if (queue.IsTesting()) + minPlayersPerTeam = 1; + + m_selectionPools[TEAM_INDEX_ALLIANCE].Init(); + m_selectionPools[TEAM_INDEX_HORDE].Init(); + + // check if there is premade against premade match + if (CheckPremadeMatch(bracketId, minPlayersPerTeam, maxPlayersPerTeam)) + { + BattleGround* bgTemplate = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); + if (!bgTemplate) + { + sLog.outError("BattleGround: CreateNewBattleGround - bg template not found for %u", bgTypeId); + return; + } + + BattleGroundInQueueInfo bgInfo; + bgInfo.Fill(bgTemplate); + bgInfo.bracketId = bracketId; + bgInfo.instanceId = sMapMgr.GenerateInstanceId(); + bgInfo.m_clientInstanceId = queue.CreateClientVisibleInstanceId(bgTypeId, bracketId); + + queue.AddBgToFreeSlots(bgInfo); + + // invite those selection pools + for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) + for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.end(); ++citr) + InviteGroupToBg((*citr), bgInfo, (*citr)->groupTeam); + + // clear structures + m_selectionPools[TEAM_INDEX_ALLIANCE].Init(); + m_selectionPools[TEAM_INDEX_HORDE].Init(); + + sWorld.GetMessager().AddMessage([instanceId = bgInfo.instanceId, clientInstanceId = bgInfo.m_clientInstanceId, bgTypeId, bracketId, allianceCount = bgInfo.GetInvitedCount(ALLIANCE), hordeCount = bgInfo.GetInvitedCount(HORDE)](World* world) + { + // create new battleground + BattleGround* bg2 = sBattleGroundMgr.CreateNewBattleGround(bgTypeId, bracketId, instanceId, clientInstanceId); + MANGOS_ASSERT(bg2); + bg2->SetInvitedCount(ALLIANCE, allianceCount); + bg2->SetInvitedCount(HORDE, hordeCount); + // start bg + bg2->StartBattleGround(); + }); + } + + // now check if there are in queues enough players to start new game of (normal battleground, or non-rated arena) + // if there are enough players in pools, start new battleground or non rated arena + if (CheckNormalMatch(queue, bgTemplate, bracketId, minPlayersPerTeam, maxPlayersPerTeam)) + { + BattleGround* bgTemplate = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); + if (!bgTemplate) + { + sLog.outError("BattleGround: CreateNewBattleGround - bg template not found for %u", bgTypeId); + return; + } + + BattleGroundInQueueInfo bgInfo; + bgInfo.Fill(bgTemplate); + bgInfo.bracketId = bracketId; + bgInfo.instanceId = sMapMgr.GenerateInstanceId(); + bgInfo.m_clientInstanceId = queue.CreateClientVisibleInstanceId(bgTypeId, bracketId); + + queue.AddBgToFreeSlots(bgInfo); + + // invite those selection pools + for (uint8 i = 0; i < PVP_TEAM_COUNT; ++i) + for (GroupsQueueType::const_iterator citr = m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.begin(); citr != m_selectionPools[TEAM_INDEX_ALLIANCE + i].selectedGroups.end(); ++citr) + InviteGroupToBg((*citr), bgInfo, (*citr)->groupTeam); + + sWorld.GetMessager().AddMessage([instanceId = bgInfo.instanceId, clientInstanceId = bgInfo.m_clientInstanceId, bgTypeId, bracketId, allianceCount = bgInfo.GetInvitedCount(ALLIANCE), hordeCount = bgInfo.GetInvitedCount(HORDE)](World* world) + { + // create new battleground + BattleGround* bg2 = sBattleGroundMgr.CreateNewBattleGround(bgTypeId, bracketId, instanceId, clientInstanceId); + MANGOS_ASSERT(bg2); + bg2->SetInvitedCount(ALLIANCE, allianceCount); + bg2->SetInvitedCount(HORDE, hordeCount); + // start bg + bg2->StartBattleGround(); + }); + } +} + +/*********************************************************/ +/*** BATTLEGROUND QUEUE EVENTS ***/ +/*********************************************************/ + +/** + Function that executes battleground queue invite event +*/ +bool BgQueueInviteEvent::Execute(uint64 /*e_time*/, uint32 /*p_time*/) +{ + sWorld.GetMessager().AddMessage([event = *this](World* /*world*/) + { + Player* plr = sObjectMgr.GetPlayer(event.m_playerGuid); + // player logged off (we should do nothing, he is correctly removed from queue in another procedure) + if (!plr) + return; + + BattleGround* bg = sBattleGroundMgr.GetBattleGround(event.m_bgInstanceGuid, event.m_bgTypeId); + // if battleground ended and its instance deleted - do nothing + if (!bg) + return; + + bool bgExists = bg; + BattleGroundTypeId bgTypeId = event.m_bgTypeId; + uint32 bgInstanceId = bg->GetClientInstanceId(); + uint32 mapId = bg->GetMapId(); + + BattleGroundQueueTypeId bgQueueTypeId = BattleGroundMgr::BgQueueTypeId(bg->GetTypeId()); + uint32 queueSlot = plr->GetBattleGroundQueueIndex(bgQueueTypeId); + if (queueSlot < PLAYER_MAX_BATTLEGROUND_QUEUES) // player is in queue or in battleground + { + sWorld.GetBGQueue().GetMessager().AddMessage([playerGuid = event.m_playerGuid, bgTypeId, bgInstanceId, mapId, removeTime = event.m_removeTime, bgQueueTypeId, queueSlot, bgExists](BattleGroundQueue* queue) + { + // check if player is invited to this bg + BattleGroundQueueItem& bgQueue = queue->m_battleGroundQueues[bgQueueTypeId]; + if (bgQueue.IsPlayerInvited(playerGuid, bgInstanceId, removeTime)) + { + WorldPacket data; + // we must send remaining time in queue + BattleGroundMgr::BuildBattleGroundStatusPacket(data, bgExists, bgTypeId, bgInstanceId, mapId, queueSlot, STATUS_WAIT_JOIN, INVITE_ACCEPT_WAIT_TIME - INVITATION_REMIND_TIME, 0); + + sWorld.GetMessager().AddMessage([playerGuid, bgQueueTypeId, data](World* /*world*/) + { + if (Player* plr = sObjectMgr.GetPlayer(playerGuid)) + { + plr->GetSession()->SendPacket(data); + } + }); + } + }); + } + }); + return true; // event will be deleted +} + +void BgQueueInviteEvent::Abort(uint64 /*e_time*/) +{ + // do nothing +} + +/** + Function that executes battleground queue remove event + this event has many possibilities when it is executed: + 1. player is in battleground ( he clicked enter on invitation window ) + 2. player left battleground queue and he isn't there any more + 3. player left battleground queue and he joined it again and IsInvitedToBGInstanceGUID = 0 + 4. player left queue and he joined again and he has been invited to same battleground again -> we should not remove him from queue yet + 5. player is invited to bg and he didn't choose what to do and timer expired - only in this condition we should call queue::RemovePlayer + we must remove player in the 5. case even if battleground object doesn't exist! +*/ +bool BgQueueRemoveEvent::Execute(uint64 /*e_time*/, uint32 /*p_time*/) +{ + sWorld.GetMessager().AddMessage([event = *this](World* /*world*/) + { + Player* plr = sObjectMgr.GetPlayer(event.m_playerGuid); + if (!plr) + // player logged off (we should do nothing, he is correctly removed from queue in another procedure) + return; + + BattleGround* bg = sBattleGroundMgr.GetBattleGround(event.m_bgInstanceGuid, event.m_bgTypeId); + // battleground can be deleted already when we are removing queue info + // bg pointer can be nullptr! so use it carefully! + + bool isBattleGround = bg; + bool bgExists = bg; + BattleGroundTypeId bgTypeId = event.m_bgTypeId; + uint32 bgInstanceId = bg ? bg->GetClientInstanceId() : 0; + uint32 mapId = bg ? bg->GetMapId() : 0; + BattleGroundStatus bgStatus = bg ? bg->GetStatus() : STATUS_NONE; + BattleGroundBracketId bracketId = bg ? bg->GetBracketId() : BG_BRACKET_ID_TEMPLATE; + + uint32 queueSlot = plr->GetBattleGroundQueueIndex(event.m_bgQueueTypeId); + if (queueSlot < PLAYER_MAX_BATTLEGROUND_QUEUES) // player is in queue, or in Battleground + { + sWorld.GetBGQueue().GetMessager().AddMessage([bgQueueTypeId = event.m_bgQueueTypeId, bgTypeId, bgStatus, playerGuid = event.m_playerGuid, instanceGuid = event.m_bgInstanceGuid, removeTime = event.m_removeTime, isBattleGround, queueSlot, bracketId, bgExists, bgInstanceId, mapId](BattleGroundQueue* queue) + { + // check if player is in queue for this BG and if we are removing his invite event + BattleGroundQueueItem& bgQueue = queue->m_battleGroundQueues[bgQueueTypeId]; + if (bgQueue.IsPlayerInvited(playerGuid, instanceGuid, removeTime)) + { + DEBUG_LOG("Battleground: removing player %u from bg queue for instance %u because of not pressing enter battle in time.", playerGuid.GetCounter(), instanceGuid); + + bgQueue.RemovePlayer(*queue, playerGuid, true); + + // update queues if battleground isn't ended + if (isBattleGround && bgStatus != STATUS_WAIT_LEAVE) + queue->ScheduleQueueUpdate(bgQueueTypeId, bgTypeId, bracketId); + + WorldPacket data; + BattleGroundMgr::BuildBattleGroundStatusPacket(data, bgExists, bgTypeId, bgInstanceId, mapId, queueSlot, STATUS_NONE, 0, 0); + + sWorld.GetMessager().AddMessage([playerGuid, bgQueueTypeId, data](World* /*world*/) + { + if (Player* plr = sObjectMgr.GetPlayer(playerGuid)) + { + plr->RemoveBattleGroundQueueId(bgQueueTypeId); + plr->GetSession()->SendPacket(data); + } + }); + } + }); + } + }); + + // event will be deleted + return true; +} + +void BgQueueRemoveEvent::Abort(uint64 /*e_time*/) +{ + // do nothing +} + +BattleGroundQueue::BattleGroundQueue() : m_testing(false) +{ + +} + +void BattleGroundQueue::Update() +{ + TimePoint previously = sWorld.GetCurrentClockTime(); + while (!World::IsStopped()) + { + TimePoint now = std::chrono::time_point_cast(Clock::now()); + GetMessager().Execute(this); + + // update scheduled queues + if (!m_queueUpdateScheduler.empty()) + { + std::vector scheduled; + { + // copy vector and clear the other + scheduled = std::vector(m_queueUpdateScheduler); + m_queueUpdateScheduler.clear(); + } + + for (unsigned long long i : scheduled) + { + BattleGroundQueueTypeId bgQueueTypeId = BattleGroundQueueTypeId(i >> 16 & 255); + BattleGroundTypeId bgTypeId = BattleGroundTypeId((i >> 8) & 255); + BattleGroundBracketId bracket_id = BattleGroundBracketId(i & 255); + + m_battleGroundQueues[bgQueueTypeId].Update(*this, bgTypeId, bracket_id); + } + } + + std::this_thread::sleep_for(std::chrono::milliseconds(500)); + }; +} + +/** + Method that schedules queue update + + @param arena rating + @param arena type + @param battleground queue type id + @param battleground type id + @param bracket id +*/ +void BattleGroundQueue::ScheduleQueueUpdate(BattleGroundQueueTypeId bgQueueTypeId, BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId) +{ + // we will use only 1 number created of bgTypeId and bracket_id + uint32 schedule_id = (bgQueueTypeId << 16) | (bgTypeId << 8) | bracketId; + bool found = false; + for (unsigned long long i : m_queueUpdateScheduler) + { + if (i == schedule_id) + { + found = true; + break; + } + } + if (!found) + m_queueUpdateScheduler.push_back(schedule_id); +} + +void BattleGroundQueue::AddBgToFreeSlots(BattleGroundInQueueInfo const& info) +{ + auto& typeIdQueue = m_bgFreeSlotQueue[info.GetTypeId()]; + typeIdQueue.emplace_front(info); +} + +void BattleGroundQueue::RemoveBgFromFreeSlots(BattleGroundTypeId typeId, uint32 instanceId) +{ + auto& typeIdQueue = m_bgFreeSlotQueue[typeId]; + for (auto itr = typeIdQueue.begin(); itr != typeIdQueue.end(); ++itr) + { + if (itr->GetInstanceId() == instanceId) + { + typeIdQueue.erase(itr); + return; + } + } +} + +BgFreeSlotQueueType& BattleGroundQueue::GetFreeSlotQueueItem(BattleGroundTypeId bgTypeId) +{ + return m_bgFreeSlotQueue[bgTypeId]; +} + +BattleGroundInQueueInfo* BattleGroundQueue::GetFreeSlotInstance(BattleGroundTypeId bgTypeId, uint32 instanceId) +{ + auto& queueItem = GetFreeSlotQueueItem(bgTypeId); + auto itr = std::find_if(queueItem.begin(), queueItem.end(), [instanceId](BattleGroundInQueueInfo const& bgInQueue) + { + return bgInQueue.instanceId == instanceId; + }); + if (itr == queueItem.end()) + return nullptr; + return &(*itr); +} + +BattleGroundQueueItem& BattleGroundQueue::GetBattleGroundQueue(BattleGroundQueueTypeId bgQueueTypeId) +{ + return m_battleGroundQueues[bgQueueTypeId]; +} + +/** + Method that builds battleground list data + + @param packet + @param battlemaster guid + @param player + @param battleground type id +*/ +void BattleGroundQueue::BuildBattleGroundListPacket(WorldPacket& data, ObjectGuid guid, uint32 playerLevel, BattleGroundTypeId bgTypeId) const +{ + uint32 mapId = GetBattleGrounMapIdByTypeId(bgTypeId); + + data.Initialize(SMSG_BATTLEFIELD_LIST); + data << guid; // battlemaster guid + data << uint32(mapId); + data << uint8(0x00); // min level offset + + // battleground - indented for cross core compatibility + { + size_t count_pos = data.wpos(); + uint32 count = 0; + data << uint32(0); // number of bg instances + + uint32 bracket_id = sBattleGroundMgr.GetBattleGroundBracketIdFromLevel(bgTypeId, playerLevel); + ClientBattleGroundIdSet const& ids = m_clientBattleGroundIds[bgTypeId][bracket_id]; + for (std::set::const_iterator itr = ids.begin(); itr != ids.end(); ++itr) + { + data << uint32(*itr); + ++count; + } + data.put(count_pos, count); + } +} + +/** + Function that returns client instance id from battleground type id and bracket id + + @param battleground type id + @param bracket id +*/ +uint32 BattleGroundQueue::CreateClientVisibleInstanceId(BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId) +{ + // we create here an instanceid, which is just for + // displaying this to the client and without any other use.. + // the client-instanceIds are unique for each battleground-type + // the instance-id just needs to be as low as possible, beginning with 1 + // the following works, because std::set is default ordered with "<" + // the optimalization would be to use as bitmask std::vector - but that would only make code unreadable + + uint32 lastId = 0; + ClientBattleGroundIdSet& ids = m_clientBattleGroundIds[bgTypeId][bracketId]; + for (ClientBattleGroundIdSet::const_iterator itr = ids.begin(); itr != ids.end();) + { + if ((++lastId) != *itr) // if there is a gap between the ids, we will break.. + break; + lastId = *itr; + } + ids.insert(lastId + 1); + + return lastId + 1; +} + +void BattleGroundQueue::RemovePlayer(BattleGroundQueueTypeId bgQueueTypeId, ObjectGuid player, bool decreaseInvitedCount) +{ + m_battleGroundQueues[bgQueueTypeId].RemovePlayer(*this, player, decreaseInvitedCount); +} + +void BattleGroundInQueueInfo::DecreaseInvitedCount(Team team) +{ + uint32 count = (team == ALLIANCE) ? --m_invitedAlliance : --m_invitedHorde; + sWorld.GetMessager().AddMessage([bgTypeId = GetTypeId(), instanceId = GetInstanceId(), team, count](World* world) + { + if (BattleGround* bg = sBattleGroundMgr.GetBattleGround(instanceId, bgTypeId)) + bg->SetInvitedCount(team, count); + }); +} + +void BattleGroundInQueueInfo::IncreaseInvitedCount(Team team) +{ + uint32 count = (team == ALLIANCE) ? ++m_invitedAlliance : ++m_invitedHorde; + sWorld.GetMessager().AddMessage([bgTypeId = GetTypeId(), instanceId = GetInstanceId(), team, count](World* world) + { + if (BattleGround* bg = sBattleGroundMgr.GetBattleGround(instanceId, bgTypeId)) + bg->SetInvitedCount(team, count); + }); +} + +void BattleGroundInQueueInfo::Fill(BattleGround* bg) +{ + bgTypeId = bg->GetTypeId(); + instanceId = bg->GetInstanceId(); + isBattleGround = true; + bracketId = bg->GetBracketId(); + status = bg->GetStatus(); + m_clientInstanceId = bg->GetClientInstanceId(); + mapId = bg->GetMapId(); + playersInside = bg->GetPlayersSize(); + maxPlayers = bg->GetMaxPlayers(); + m_invitedAlliance = bg->GetInvitedCount(ALLIANCE); + m_invitedHorde = bg->GetInvitedCount(HORDE); + m_maxPlayersPerTeam = bg->GetMaxPlayersPerTeam(); + m_minPlayersPerTeam = bg->GetMinPlayersPerTeam(); +} diff --git a/src/game/BattleGround/BattleGroundQueue.h b/src/game/BattleGround/BattleGroundQueue.h new file mode 100644 index 0000000000..56307264c0 --- /dev/null +++ b/src/game/BattleGround/BattleGroundQueue.h @@ -0,0 +1,273 @@ +/* + * This file is part of the CMaNGOS Project. See AUTHORS file for Copyright information + * + * 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 + * (at your option) 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, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#ifndef __BATTLEGROUNDQUEUE_H +#define __BATTLEGROUNDQUEUE_H + +#include "Common.h" +#include "BattleGround/BattleGround.h" + +struct GroupQueueInfo; // type predefinition +struct PlayerQueueInfo // stores information for players in queue +{ + uint32 lastOnlineTime; // for tracking and removing offline players from queue after 5 minutes + GroupQueueInfo* groupInfo; // pointer to the associated groupqueueinfo +}; + +typedef std::map GroupQueueInfoPlayers; + +struct AddGroupToQueueInfo +{ + Team team; + std::vector members; // empty when not in group + uint32 clientInstanceId; + uint32 mapId; +}; + +struct GroupQueueInfo // stores information about the group in queue (also used when joined as solo!) +{ + GroupQueueInfoPlayers players; // player queue info map + Team groupTeam; // Player team (ALLIANCE/HORDE) + BattleGroundTypeId bgTypeId; // battleground type id + BattleGroundBracketId bgBracketId; // battleground bracked id + uint32 mapId; // invited to mapId + uint32 clientInstanceId; // invited to client instance id + uint32 joinTime; // time when group was added + uint32 removeInviteTime; // time when we will remove invite for players in group + uint32 isInvitedToBgInstanceGuid; // was invited to certain BG + uint32 desiredInstanceId; // queued for this instance specifically +}; + +struct BattleGroundInQueueInfo +{ + BattleGroundTypeId bgTypeId; + uint32 instanceId; + bool isBattleGround; + BattleGroundBracketId bracketId; + BattleGroundStatus status; // only altered in bg and synched here + uint32 m_clientInstanceId; + uint32 mapId; + + uint32 playersInside; + uint32 maxPlayers; + uint32 m_maxPlayersPerTeam; + uint32 m_minPlayersPerTeam; + + uint32 GetInstanceId() const { return instanceId; } + bool IsBattleGround() const { return isBattleGround; } + BattleGroundTypeId GetTypeId() const { return bgTypeId; } + BattleGroundBracketId GetBracketId() const { return bracketId; } + BattleGroundStatus GetStatus() const { return status; } + uint32 GetClientInstanceId() const { return m_clientInstanceId; } + uint32 GetMapId() const { return mapId; } + + uint32 GetPlayersSize() const { return playersInside; } + uint32 GetMaxPlayers() const { return maxPlayers; } + bool HasFreeSlots() const { return GetPlayersSize() < GetMaxPlayers(); } + uint32 GetMaxPlayersPerTeam() const { return m_maxPlayersPerTeam; } + uint32 GetMinPlayersPerTeam() const { return m_minPlayersPerTeam; } + + void DecreaseInvitedCount(Team team); + void IncreaseInvitedCount(Team team); + uint32 GetInvitedCount(Team team) const + { + if (team == ALLIANCE) + return m_invitedAlliance; + return m_invitedHorde; + } + + uint32 GetFreeSlotsForTeam(Team team) const + { + // return free slot count to MaxPlayerPerTeam + if (GetStatus() == STATUS_WAIT_JOIN || GetStatus() == STATUS_IN_PROGRESS) + return (GetInvitedCount(team) < GetMaxPlayersPerTeam()) ? GetMaxPlayersPerTeam() - GetInvitedCount(team) : 0; + + return 0; + } + + void Fill(BattleGround* bg); + + private: + uint32 m_invitedAlliance; + uint32 m_invitedHorde; +}; + +// this container can't be deque, because deque doesn't like removing the last element - if you remove it, it invalidates next iterator and crash appears +typedef std::list BgFreeSlotQueueType; + +class BattleGroundQueue; + +class BattleGroundQueueItem +{ + public: + BattleGroundQueueItem(); + ~BattleGroundQueueItem(); + + void Update(BattleGroundQueue& queue, BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketId*/); + + void FillPlayersToBg(BattleGroundInQueueInfo& queueInfo, BattleGroundBracketId /*bracketId*/); + bool CheckPremadeMatch(BattleGroundBracketId /*bracketId*/, uint32 /*minPlayersPerTeam*/, uint32 /*maxPlayersPerTeam*/); + bool CheckNormalMatch(BattleGroundQueue& queue, BattleGround* /*bgTemplate*/, BattleGroundBracketId /*bracketId*/, uint32 /*minPlayers*/, uint32 /*maxPlayers*/); + bool CheckSkirmishForSameFaction(BattleGroundBracketId /*bracketId*/, uint32 /*minPlayersPerTeam*/); + GroupQueueInfo* AddGroup(ObjectGuid leader, AddGroupToQueueInfo const& /*groupInfo*/, BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketEntry*/, bool /*isPremade*/, uint32 /*instanceId*/); + void RemovePlayer(BattleGroundQueue& queue, ObjectGuid guid, bool decreaseInvitedCount); + bool IsPlayerInvited(ObjectGuid /*playerGuid*/, const uint32 /*bgInstanceGuid*/, const uint32 /*removeTime*/); + bool GetPlayerGroupInfoData(ObjectGuid /*guid*/, GroupQueueInfo* /*groupInfo*/); + void PlayerInvitedToBgUpdateAverageWaitTime(GroupQueueInfo* /*groupInfo*/, BattleGroundBracketId /*bracketId*/); + uint32 GetAverageQueueWaitTime(GroupQueueInfo* /*groupInfo*/, BattleGroundBracketId /*bracketId*/); + + private: + typedef std::map QueuedPlayersMap; + QueuedPlayersMap m_queuedPlayers; + + // we need constant add to begin and constant remove / add from the end, therefore deque suits our problem well + typedef std::list GroupsQueueType; + + /* + This two dimensional array is used to store All queued groups + First dimension specifies the bgTypeId + Second dimension specifies the player's group types - + BG_QUEUE_PREMADE_ALLIANCE is used for premade alliance groups and alliance rated arena teams + BG_QUEUE_PREMADE_HORDE is used for premade horde groups and horde rated arena teams + BG_QUEUE_NORMAL_ALLIANCE is used for normal (or small) alliance groups or non-rated arena matches + BG_QUEUE_NORMAL_HORDE is used for normal (or small) horde groups or non-rated arena matches + */ + GroupsQueueType m_queuedGroups[MAX_BATTLEGROUND_BRACKETS][BG_QUEUE_GROUP_TYPES_COUNT]; + + // class to select and invite groups to bg + class SelectionPool + { + public: + SelectionPool() : playerCount(0) {} + void Init(); + bool AddGroup(GroupQueueInfo* ginfo, uint32 desiredCount, uint32 bgInstanceId); + bool KickGroup(uint32 size); + uint32 GetPlayerCount() const { return playerCount; } + GroupsQueueType selectedGroups; + private: + uint32 playerCount; + }; + + // one selection pool for horde, other one for alliance + SelectionPool m_selectionPools[PVP_TEAM_COUNT]; + + bool InviteGroupToBg(GroupQueueInfo* groupInfo, BattleGroundInQueueInfo& queueInfo, Team side); + + uint32 m_waitTimes[PVP_TEAM_COUNT][MAX_BATTLEGROUND_BRACKETS][COUNT_OF_PLAYERS_TO_AVERAGE_WAIT_TIME]; + uint32 m_waitTimeLastPlayer[PVP_TEAM_COUNT][MAX_BATTLEGROUND_BRACKETS]; + uint32 m_sumOfWaitTimes[PVP_TEAM_COUNT][MAX_BATTLEGROUND_BRACKETS]; +}; + +/* + This class is used to invite player to BG again, when minute lasts from his first invitation + it is capable to solve all possibilities +*/ +class BgQueueInviteEvent : public BasicEvent +{ + public: + BgQueueInviteEvent(ObjectGuid playerGuid, uint32 bgInstanceGuid, BattleGroundTypeId bgTypeId, uint32 removeTime) : + m_playerGuid(playerGuid), m_bgInstanceGuid(bgInstanceGuid), m_bgTypeId(bgTypeId), m_removeTime(removeTime) + { + }; + virtual ~BgQueueInviteEvent() {}; + + virtual bool Execute(uint64 e_time, uint32 p_time) override; + virtual void Abort(uint64 e_time) override; + + private: + ObjectGuid m_playerGuid; + uint32 m_bgInstanceGuid; + BattleGroundTypeId m_bgTypeId; + uint32 m_removeTime; +}; + +/* + This class is used to remove player from BG queue after 1 minute 20 seconds from first invitation + We must store removeInvite time in case player left queue and joined and is invited again + We must store bgQueueTypeId, because battleground can be deleted already, when player entered it +*/ +class BgQueueRemoveEvent : public BasicEvent +{ + public: + BgQueueRemoveEvent(ObjectGuid playerGuid, uint32 bgInstanceGuid, BattleGroundTypeId bgTypeId, BattleGroundQueueTypeId bgQueueTypeId, uint32 removeTime) + : m_playerGuid(playerGuid), m_bgInstanceGuid(bgInstanceGuid), m_removeTime(removeTime), m_bgTypeId(bgTypeId), m_bgQueueTypeId(bgQueueTypeId) + {} + + virtual ~BgQueueRemoveEvent() {} + + virtual bool Execute(uint64 e_time, uint32 p_time) override; + virtual void Abort(uint64 e_time) override; + + private: + ObjectGuid m_playerGuid; + uint32 m_bgInstanceGuid; + uint32 m_removeTime; + BattleGroundTypeId m_bgTypeId; + BattleGroundQueueTypeId m_bgQueueTypeId; +}; + +class BattleGroundQueue +{ + friend class BgQueueInviteEvent; + friend class BgQueueRemoveEvent; + public: + BattleGroundQueue(); + + void Update(); + + Messager& GetMessager() { return m_messager; } + + void ScheduleQueueUpdate(BattleGroundQueueTypeId /*bgQueueTypeId*/, BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketId*/); + + void AddBgToFreeSlots(BattleGroundInQueueInfo const& info); + void RemoveBgFromFreeSlots(BattleGroundTypeId typeId, uint32 instanceId); + + void RemovePlayer(BattleGroundQueueTypeId bgQueueTypeId, ObjectGuid player, bool decreaseInvitedCount); + + BgFreeSlotQueueType& GetFreeSlotQueueItem(BattleGroundTypeId bgTypeId); + BattleGroundInQueueInfo* GetFreeSlotInstance(BattleGroundTypeId bgTypeId, uint32 instanceId); + BattleGroundQueueItem& GetBattleGroundQueue(BattleGroundQueueTypeId bgQueueTypeId); + + void SetNextRatingDiscardUpdate(std::chrono::milliseconds timePoint); + + bool IsTesting() const { return m_testing; } + void SetTesting(bool state) { m_testing = state; } + + uint32 CreateClientVisibleInstanceId(BattleGroundTypeId /*bgTypeId*/, BattleGroundBracketId /*bracketId*/); + void DeleteClientVisibleInstanceId(BattleGroundTypeId bgTypeId, BattleGroundBracketId bracketId, uint32 clientInstanceId) + { + m_clientBattleGroundIds[bgTypeId][bracketId].erase(clientInstanceId); + } + + void BuildBattleGroundListPacket(WorldPacket& data, ObjectGuid guid, uint32 playerLevel, BattleGroundTypeId bgTypeId) const; + private: + BattleGroundQueueItem m_battleGroundQueues[MAX_BATTLEGROUND_QUEUE_TYPES]; + + BgFreeSlotQueueType m_bgFreeSlotQueue[MAX_BATTLEGROUND_TYPE_ID]; + + std::vector m_queueUpdateScheduler; + + Messager m_messager; + + bool m_testing; + + typedef std::set ClientBattleGroundIdSet; + ClientBattleGroundIdSet m_clientBattleGroundIds[MAX_BATTLEGROUND_TYPE_ID][MAX_BATTLEGROUND_BRACKETS]; // the instanceids just visible for the client +}; + +#endif \ No newline at end of file diff --git a/src/game/Chat/Level3.cpp b/src/game/Chat/Level3.cpp index e9be18eefe..9f87b396ca 100644 --- a/src/game/Chat/Level3.cpp +++ b/src/game/Chat/Level3.cpp @@ -763,7 +763,10 @@ bool ChatHandler::HandleReloadItemRequiredTragetCommand(char* /*args*/) bool ChatHandler::HandleReloadBattleEventCommand(char* /*args*/) { sLog.outString("Re-Loading BattleGround Eventindexes..."); - sBattleGroundMgr.LoadBattleEventIndexes(); + sBattleGroundMgr.GetMessager().AddMessage([](BattleGroundMgr* mgr) + { + mgr->LoadBattleEventIndexes(true); + }); SendGlobalSysMessage("DB table `gameobject_battleground` and `creature_battleground` reloaded."); return true; } diff --git a/src/game/Chat/debugcmds.cpp b/src/game/Chat/debugcmds.cpp index 362d952a7c..d6337d862e 100644 --- a/src/game/Chat/debugcmds.cpp +++ b/src/game/Chat/debugcmds.cpp @@ -723,7 +723,10 @@ bool ChatHandler::HandleDebugGetItemStateCommand(char* args) bool ChatHandler::HandleDebugBattlegroundCommand(char* /*args*/) { - sBattleGroundMgr.ToggleTesting(); + sBattleGroundMgr.GetMessager().AddMessage([](BattleGroundMgr* mgr) + { + mgr->ToggleTesting(); + }); return true; } diff --git a/src/game/Entities/Creature.cpp b/src/game/Entities/Creature.cpp index 6417ed4aad..eae5fd52e3 100644 --- a/src/game/Entities/Creature.cpp +++ b/src/game/Entities/Creature.cpp @@ -660,7 +660,7 @@ uint32 Creature::ChooseDisplayId(const CreatureInfo* cinfo, const CreatureData* { if (cinfo->DisplayId[i]) { - if (roll < cinfo->DisplayIdProbability[i]) + if (roll < int32(cinfo->DisplayIdProbability[i])) { display_id = cinfo->DisplayId[i]; break; @@ -1015,7 +1015,7 @@ bool Creature::CanInteractWithBattleMaster(Player* pPlayer, bool msg) const if (!isBattleMaster()) return false; - BattleGroundTypeId bgTypeId = sBattleGroundMgr.GetBattleMasterBG(GetEntry()); + BattleGroundTypeId bgTypeId = GetMap()->GetMapDataContainer().GetBattleMasterBG(GetEntry()); if (bgTypeId == BATTLEGROUND_TYPE_NONE) return false; diff --git a/src/game/Entities/Player.cpp b/src/game/Entities/Player.cpp index 0dd8c20a75..95be3184ad 100644 --- a/src/game/Entities/Player.cpp +++ b/src/game/Entities/Player.cpp @@ -18308,41 +18308,6 @@ bool Player::GetBGAccessByLevel(BattleGroundTypeId bgTypeId) const return true; } -uint32 Player::GetMinLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId) -{ - if (bracket_id < 1) - return 0; - - if (bracket_id > BG_BRACKET_ID_LAST) - bracket_id = BG_BRACKET_ID_LAST; - - BattleGround* bg = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); - assert(bg); - return 10 * bracket_id + bg->GetMinLevel(); -} - -uint32 Player::GetMaxLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId) -{ - if (bracket_id >= BG_BRACKET_ID_LAST) - return 255; // hardcoded max level - - return GetMinLevelForBattleGroundBracketId(bracket_id, bgTypeId) + 10; -} - -BattleGroundBracketId Player::GetBattleGroundBracketIdFromLevel(BattleGroundTypeId bgTypeId) const -{ - BattleGround* bg = sBattleGroundMgr.GetBattleGroundTemplate(bgTypeId); - assert(bg); - if (GetLevel() < bg->GetMinLevel()) - return BG_BRACKET_ID_FIRST; - - uint32 bracket_id = (GetLevel() - bg->GetMinLevel()) / 10; - if (bracket_id > MAX_BATTLEGROUND_BRACKETS) - return BG_BRACKET_ID_LAST; - - return BattleGroundBracketId(bracket_id); -} - float Player::GetReputationPriceDiscount(Creature const* creature) const { return GetReputationPriceDiscount(creature->GetFactionTemplateEntry()); diff --git a/src/game/Entities/Player.h b/src/game/Entities/Player.h index f622d4c1f8..0fa10ace5e 100644 --- a/src/game/Entities/Player.h +++ b/src/game/Entities/Player.h @@ -41,6 +41,7 @@ #include "Server/SQLStorages.h" #include "Loot/LootMgr.h" #include "Cinematics/CinematicMgr.h" +#include "BattleGround/BattleGroundDefines.h" #include @@ -1877,10 +1878,6 @@ class Player : public Unit BattleGroundTypeId GetBattleGroundTypeId() const { return m_bgData.bgTypeID; } BattleGround* GetBattleGround() const; - static uint32 GetMinLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId); - static uint32 GetMaxLevelForBattleGroundBracketId(BattleGroundBracketId bracket_id, BattleGroundTypeId bgTypeId); - BattleGroundBracketId GetBattleGroundBracketIdFromLevel(BattleGroundTypeId bgTypeId) const; - bool InBattleGroundQueue() const { for (auto i : m_bgBattleGroundQueueID) diff --git a/src/game/Globals/SharedDefines.h b/src/game/Globals/SharedDefines.h index 5480621992..610467fead 100644 --- a/src/game/Globals/SharedDefines.h +++ b/src/game/Globals/SharedDefines.h @@ -1864,41 +1864,6 @@ enum BanReturn BAN_NOTFOUND }; -// indexes of BattlemasterList.dbc -enum BattleGroundTypeId -{ - BATTLEGROUND_TYPE_NONE = 0, - BATTLEGROUND_AV = 1, - BATTLEGROUND_WS = 2, - BATTLEGROUND_AB = 3, -}; -#define MAX_BATTLEGROUND_TYPE_ID 4 - -inline BattleGroundTypeId GetBattleGroundTypeIdByMapId(uint32 mapId) -{ - switch (mapId) - { - case 30: return BATTLEGROUND_AV; - case 489: return BATTLEGROUND_WS; - case 529: return BATTLEGROUND_AB; - default: return BATTLEGROUND_TYPE_NONE; - } -} - -inline uint32 GetBattleGrounMapIdByTypeId(BattleGroundTypeId bgTypeId) -{ - switch (bgTypeId) - { - case BATTLEGROUND_AV: return 30; - case BATTLEGROUND_WS: return 489; - case BATTLEGROUND_AB: return 529; - default: return 0; // none - } - - // impossible, just make compiler happy - return 0; -} - enum MailResponseType { MAIL_SEND = 0, diff --git a/src/game/Grids/GridNotifiers.cpp b/src/game/Grids/GridNotifiers.cpp index d5108c1e79..91f40fcf2f 100644 --- a/src/game/Grids/GridNotifiers.cpp +++ b/src/game/Grids/GridNotifiers.cpp @@ -209,7 +209,7 @@ void MaNGOS::RespawnDo::operator()(Creature* u) const Map* map = u->GetMap(); if (map->IsBattleGround()) { - BattleGroundEventIdx eventId = sBattleGroundMgr.GetCreatureEventIndex(u->GetDbGuid()); + BattleGroundEventIdx eventId = map->GetMapDataContainer().GetCreatureEventIndex(u->GetDbGuid()); if (!((BattleGroundMap*)map)->GetBG()->IsActiveEvent(eventId.event1, eventId.event2)) return; } @@ -234,8 +234,8 @@ void MaNGOS::RespawnDo::operator()(GameObject* u) const Map* map = u->GetMap(); if (map->IsBattleGround()) { - BattleGroundEventIdx eventId = sBattleGroundMgr.GetGameObjectEventIndex(u->GetDbGuid()); - if (!((BattleGroundMap*)map)->GetBG()->IsActiveEvent(eventId.event1, eventId.event2)) + BattleGroundEventIdx eventId = map->GetMapDataContainer().GetGameObjectEventIndex(u->GetDbGuid()); + if (!static_cast(map)->GetBG()->IsActiveEvent(eventId.event1, eventId.event2)) return; } diff --git a/src/game/Groups/Group.cpp b/src/game/Groups/Group.cpp index 796a23edf4..036bfccaed 100644 --- a/src/game/Groups/Group.cpp +++ b/src/game/Groups/Group.cpp @@ -31,8 +31,8 @@ #include "BattleGround/BattleGround.h" #include "Maps/MapManager.h" #include "Maps/MapPersistentStateMgr.h" -#include "LFG/LFGMgr.h" -#include "LFG/LFGQueue.h" +#include "Spells/SpellAuras.h" +#include "BattleGround/BattleGroundMgr.h" #ifdef BUILD_DEPRECATED_PLAYERBOT #include "PlayerBot/Base/PlayerbotMgr.h" #endif @@ -1265,7 +1265,7 @@ uint32 Group::CanJoinBattleGroundQueue(BattleGroundTypeId bgTypeId, BattleGround if (!reference) return BG_JOIN_ERR_OFFLINE_MEMBER; - BattleGroundBracketId bracket_id = reference->GetBattleGroundBracketIdFromLevel(bgTypeId); + BattleGroundBracketId bracket_id = sBattleGroundMgr.GetBattleGroundBracketIdFromLevel(bgTypeId, reference->GetLevel()); Team team = reference->GetTeam(); // check every member of the group to be able to join @@ -1279,7 +1279,7 @@ uint32 Group::CanJoinBattleGroundQueue(BattleGroundTypeId bgTypeId, BattleGround if (member->GetTeam() != team) return BG_JOIN_ERR_MIXED_FACTION; // not in the same battleground level bracket, don't let join - if (member->GetBattleGroundBracketIdFromLevel(bgTypeId) != bracket_id) + if (sBattleGroundMgr.GetBattleGroundBracketIdFromLevel(bgTypeId, member->GetLevel()) != bracket_id) return BG_JOIN_ERR_MIXED_LEVELS; // don't let join if someone from the group is already in that bg queue if (member->InBattleGroundQueueForBattleGroundQueueType(bgQueueTypeId)) diff --git a/src/game/Maps/Map.cpp b/src/game/Maps/Map.cpp index 5a52885db7..2e4de48509 100644 --- a/src/game/Maps/Map.cpp +++ b/src/game/Maps/Map.cpp @@ -2055,6 +2055,9 @@ void BattleGroundMap::Update(const uint32& diff) { Map::Update(diff); + if (!m_bg) + return; + if (!m_bg->GetPlayersSize()) { // BG is empty @@ -2069,7 +2072,10 @@ void BattleGroundMap::Update(const uint32& diff) // BattleGround Template instance cannot be updated, because it would be deleted if (!m_bg->GetInvitedCount(HORDE) && !m_bg->GetInvitedCount(ALLIANCE)) { - sBattleGroundMgr.RemoveBattleGround(GetInstanceId(), m_bg->GetTypeId()); + sBattleGroundMgr.GetMessager().AddMessage([instanceId = GetInstanceId(), typeId = m_bg->GetTypeId()](BattleGroundMgr* mgr) + { + mgr->RemoveBattleGround(instanceId, typeId); + }); m_bg = nullptr; } } diff --git a/src/game/Maps/MapDataContainer.cpp b/src/game/Maps/MapDataContainer.cpp index 652a3ec715..63c5bc48b3 100644 --- a/src/game/Maps/MapDataContainer.cpp +++ b/src/game/Maps/MapDataContainer.cpp @@ -24,12 +24,15 @@ #include "Globals/UnitCondition.h" #include "World/WorldStateExpression.h" #include "Globals/CombatCondition.h" +#include "BattleGround/BattleGroundMgr.h" MapDataContainer::MapDataContainer() : m_spellListContainer(sObjectMgr.GetCreatureSpellListContainer()), m_spawnGroupContainer(sObjectMgr.GetSpawnGroupContainer()), m_CreatureEventAIEventEntryMap(sEventAIMgr.GetCreatureEventEntryAIMap()), m_CreatureEventAIEventGuidMap(sEventAIMgr.GetCreatureEventGuidAIMap()), m_creatureEventAIComputedDataMap(sEventAIMgr.GetEAIComputedDataMap()), m_stringIds(sScriptMgr.GetStringIdMap()), m_stringIdsByString(sScriptMgr.GetStringIdByStringMap()), - m_unitConditions(sObjectMgr.GetUnitConditions()), m_worldStateExpressions(sObjectMgr.GetWorldStateExpressions()), m_combatConditions(sObjectMgr.GetCombatConditions()) + m_unitConditions(sObjectMgr.GetUnitConditions()), m_worldStateExpressions(sObjectMgr.GetWorldStateExpressions()), m_combatConditions(sObjectMgr.GetCombatConditions()), + m_creatureBattleEventIndexMap(sBattleGroundMgr.GetCreatureEventIndexes()), m_gameObjectBattleEventIndexMap(sBattleGroundMgr.GetGameObjectEventIndexes()), + m_battleMastersMap(sBattleGroundMgr.GetBattleMastersMap()) { for (uint32 i = 0; i < SCRIPT_TYPE_MAX; ++i) SetScriptMap(ScriptMapType(i), sScriptMgr.GetScriptMap(ScriptMapType(i))); @@ -143,6 +146,48 @@ std::shared_ptr> MapDataContainer::GetComb return m_combatConditions; } +const BattleGroundEventIdx MapDataContainer::GetCreatureEventIndex(uint32 dbGuid) const +{ + CreatureBattleEventIndexesMap::const_iterator itr = m_creatureBattleEventIndexMap->find(dbGuid); + if (itr != m_creatureBattleEventIndexMap->end()) + return itr->second; + + return m_creatureBattleEventIndexMap->find(static_cast(-1))->second; +} + +void MapDataContainer::SetCreatureEventIndexes(std::shared_ptr indexes) +{ + m_creatureBattleEventIndexMap = indexes; +} + +const BattleGroundEventIdx MapDataContainer::GetGameObjectEventIndex(uint32 dbGuid) const +{ + GameObjectBattleEventIndexesMap::const_iterator itr = m_gameObjectBattleEventIndexMap->find(dbGuid); + if (itr != m_gameObjectBattleEventIndexMap->end()) + return itr->second; + + return m_gameObjectBattleEventIndexMap->find(static_cast(-1))->second; +} + +void MapDataContainer::SetGameObjectEventIndexes(std::shared_ptr indexes) +{ + m_gameObjectBattleEventIndexMap = indexes; +} + +BattleGroundTypeId MapDataContainer::GetBattleMasterBG(uint32 entry) const +{ + BattleMastersMap::const_iterator itr = m_battleMastersMap->find(entry); + if (itr != m_battleMastersMap->end()) + return itr->second; + + return BATTLEGROUND_TYPE_NONE; +} + +void MapDataContainer::SetBattleMastersMap(std::shared_ptr battleMasters) +{ + m_battleMastersMap = battleMasters; +} + void MapDataContainer::SetStringIdMaps(std::shared_ptr stringIds, std::shared_ptr stringIdsByString) { m_stringIds = stringIds; diff --git a/src/game/Maps/MapDataContainer.h b/src/game/Maps/MapDataContainer.h index cd25db6e4d..8f7f46cc2a 100644 --- a/src/game/Maps/MapDataContainer.h +++ b/src/game/Maps/MapDataContainer.h @@ -21,6 +21,7 @@ #include "Platform/Define.h" #include "DBScripts/ScriptMgrDefines.h" +#include "BattleGround/BattleGroundDefines.h" #include #include #include @@ -36,6 +37,7 @@ struct ScriptInfo; struct UnitConditionEntry; struct WorldStateExpressionEntry; struct CombatConditionEntry; +struct BattleGroundEventIdx; // Event_Map typedef std::vector CreatureEventAI_Event_Vec; @@ -47,6 +49,11 @@ typedef std::multimap < uint32 /*delay*/, std::shared_ptr> ScriptMap typedef std::map < uint32 /*id*/, ScriptMap > ScriptMapMap; typedef std::pair ScriptMapMapName; +// Battleground +typedef std::unordered_map BattleMastersMap; +typedef std::unordered_map CreatureBattleEventIndexesMap; +typedef std::unordered_map GameObjectBattleEventIndexesMap; + class MapDataContainer { public: @@ -74,6 +81,14 @@ class MapDataContainer std::shared_ptr> GetUnitConditions() const; std::shared_ptr> GetWorldStateExpressions() const; std::shared_ptr> GetCombatConditions() const; + + const BattleGroundEventIdx GetCreatureEventIndex(uint32 dbGuid) const; + void SetCreatureEventIndexes(std::shared_ptr indexes); + const BattleGroundEventIdx GetGameObjectEventIndex(uint32 dbGuid) const; + void SetGameObjectEventIndexes(std::shared_ptr indexes); + + BattleGroundTypeId GetBattleMasterBG(uint32 entry) const; + void SetBattleMastersMap(std::shared_ptr battleMasters); private: std::shared_ptr m_spellListContainer; std::shared_ptr m_spawnGroupContainer; @@ -91,6 +106,10 @@ class MapDataContainer std::shared_ptr> m_unitConditions; std::shared_ptr> m_worldStateExpressions; std::shared_ptr> m_combatConditions; + + std::shared_ptr m_creatureBattleEventIndexMap; + std::shared_ptr m_gameObjectBattleEventIndexMap; + std::shared_ptr m_battleMastersMap; }; #endif \ No newline at end of file diff --git a/src/game/Maps/MapManager.cpp b/src/game/Maps/MapManager.cpp index a9c79a105b..4e863b9e0a 100644 --- a/src/game/Maps/MapManager.cpp +++ b/src/game/Maps/MapManager.cpp @@ -27,6 +27,7 @@ #include "Grids/CellImpl.h" #include "Globals/ObjectMgr.h" #include "Maps/MapWorkers.h" +#include "BattleGround/BattleGroundMgr.h" #include #define CLASS_LOCK MaNGOS::ClassLevelLockable @@ -147,12 +148,12 @@ Map* MapManager::CreateMap(uint32 id, const WorldObject* obj) return m; } -Map* MapManager::CreateBgMap(uint32 mapid, BattleGround* bg) +Map* MapManager::CreateBgMap(uint32 mapid, uint32 instanceId, BattleGround* bg) { sTerrainMgr.LoadTerrain(mapid); Guard _guard(*this); - return CreateBattleGroundMap(mapid, sMapMgr.GenerateInstanceId(), bg); + return CreateBattleGroundMap(mapid, instanceId, bg); } Map* MapManager::FindMap(uint32 mapid, uint32 instanceId) const diff --git a/src/game/Maps/MapManager.h b/src/game/Maps/MapManager.h index b09db5b15a..fd43730195 100644 --- a/src/game/Maps/MapManager.h +++ b/src/game/Maps/MapManager.h @@ -64,7 +64,7 @@ class MapManager : public MaNGOS::Singleton i_MaxInstanceId; MapUpdater m_updater; }; diff --git a/src/game/Server/WorldSession.cpp b/src/game/Server/WorldSession.cpp index acd8971e29..ee38559129 100644 --- a/src/game/Server/WorldSession.cpp +++ b/src/game/Server/WorldSession.cpp @@ -688,7 +688,10 @@ void WorldSession::LogoutPlayer() if (BattleGroundQueueTypeId bgQueueTypeId = _player->GetBattleGroundQueueTypeId(i)) { _player->RemoveBattleGroundQueueId(bgQueueTypeId); - sBattleGroundMgr.m_battleGroundQueues[ bgQueueTypeId ].RemovePlayer(_player->GetObjectGuid(), true); + sWorld.GetBGQueue().GetMessager().AddMessage([bgQueueTypeId, playerGuid = _player->GetObjectGuid()](BattleGroundQueue* queue) + { + queue->RemovePlayer(bgQueueTypeId, playerGuid, true); + }); } } diff --git a/src/game/Server/WorldSession.h b/src/game/Server/WorldSession.h index 5f89683edb..2868854239 100644 --- a/src/game/Server/WorldSession.h +++ b/src/game/Server/WorldSession.h @@ -31,6 +31,7 @@ #include "Entities/Item.h" #include "Server/WorldSocket.h" #include "Multithreading/Messager.h" +#include "BattleGround/BattleGroundDefines.h" #include #include diff --git a/src/game/World/World.cpp b/src/game/World/World.cpp index 1cbc908704..d59eefb1ea 100644 --- a/src/game/World/World.cpp +++ b/src/game/World/World.cpp @@ -162,6 +162,7 @@ World::~World() MMAP::MMapFactory::clear(); m_lfgQueueThread.join(); + m_bgQueueThread.join(); } /// Cleanups before world stop @@ -1236,10 +1237,10 @@ void World::SetInitialWorldSettings() sObjectMgr.LoadGameObjectForQuests(); sLog.outString("Loading BattleMasters..."); - sBattleGroundMgr.LoadBattleMastersEntry(); + sBattleGroundMgr.LoadBattleMastersEntry(false); sLog.outString("Loading BattleGround event indexes..."); - sBattleGroundMgr.LoadBattleEventIndexes(); + sBattleGroundMgr.LoadBattleEventIndexes(false); sLog.outString("Loading GameTeleports..."); sObjectMgr.LoadGameTele(); @@ -2030,14 +2031,6 @@ bool World::RemoveBanAccount(BanMode mode, const std::string& source, const std: return true; } -void World::StartLFGQueueThread() -{ - m_lfgQueueThread = std::thread([&]() - { - m_lfgQueue.Update(); - }); -} - /// Update the game time void World::_UpdateGameTime() { @@ -2722,3 +2715,19 @@ void World::LoadWorldSafeLocs() const sWorldSafeLocsStore.Load(true); sLog.outString(">> Loaded %u world safe locs", sWorldSafeLocsStore.GetRecordCount()); } + +void World::StartLFGQueueThread() +{ + m_lfgQueueThread = std::thread([&]() + { + m_lfgQueue.Update(); + }); +} + +void World::StartBGQueueThread() +{ + m_bgQueueThread = std::thread([&]() + { + m_bgQueue.Update(); + }); +} diff --git a/src/game/World/World.h b/src/game/World/World.h index 2f903c10e6..0b77f9511e 100644 --- a/src/game/World/World.h +++ b/src/game/World/World.h @@ -31,6 +31,7 @@ #include "Multithreading/Messager.h" #include "Globals/GraveyardManager.h" #include "LFG/LFGQueue.h" +#include "BattleGround/BattleGroundQueue.h" #include #include @@ -600,7 +601,9 @@ class World void SendGMTextFlags(uint32 accountFlag, int32 stringId, std::string type, const char* message); LFGQueue& GetLFGQueue() { return m_lfgQueue; } + BattleGroundQueue& GetBGQueue() { return m_bgQueue; } void StartLFGQueueThread(); + void StartBGQueueThread(); protected: void _UpdateGameTime(); // callback for UpdateRealmCharacters @@ -720,9 +723,11 @@ class World GraveyardManager m_graveyardManager; - // Housing this here but logically it is completely asynchronous - TODO: Separate this and unify with BG queue + // Housing this here but logically it is completely asynchronous LFGQueue m_lfgQueue; std::thread m_lfgQueueThread; + BattleGroundQueue m_bgQueue; + std::thread m_bgQueueThread; }; extern uint32 realmID; diff --git a/src/mangosd/Master.cpp b/src/mangosd/Master.cpp index 69b4cf7dd7..a8334d7b2c 100644 --- a/src/mangosd/Master.cpp +++ b/src/mangosd/Master.cpp @@ -150,6 +150,7 @@ int Master::Run() } sWorld.StartLFGQueueThread(); + sWorld.StartBGQueueThread(); MaNGOS::Thread* cliThread = nullptr;