From f758b14b482903b8dc4ee13b0321a8f8e7605381 Mon Sep 17 00:00:00 2001 From: trickerer Date: Tue, 7 May 2024 13:00:45 +0700 Subject: [PATCH] NPCBots: Fix equipment not being returned and items not being retrievable from mail at bot's expiration --- src/server/game/AI/NpcBots/bot_ai.cpp | 59 ++++++++++++++++++--------- src/server/game/AI/NpcBots/bot_ai.h | 1 + 2 files changed, 41 insertions(+), 19 deletions(-) diff --git a/src/server/game/AI/NpcBots/bot_ai.cpp b/src/server/game/AI/NpcBots/bot_ai.cpp index 37162cf47ccb74..6298983f46bd72 100644 --- a/src/server/game/AI/NpcBots/bot_ai.cpp +++ b/src/server/game/AI/NpcBots/bot_ai.cpp @@ -466,7 +466,7 @@ void bot_ai::CheckOwnerExpiry() if (!BotMgr::GetOwnershipExpireTime()) return; //disabled - if (IsTempBot()) + if (IsTempBot() || !IAmFree()) return; NpcBotData const* npcBotData = BotDataMgr::SelectNpcBotData(me->GetEntry()); @@ -512,7 +512,7 @@ void bot_ai::CheckOwnerExpiry() // "FROM item_instance WHERE guid IN (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", CONNECTION_SYNCH for (uint8 i = 0; i != BOT_INVENTORY_SIZE; ++i) - stmt->SetData(i, npcBotData->equips[i]); + stmt->SetData(i, npcBotData->equips[i] ? npcBotData->equips[i] : std::numeric_limits::max()); PreparedQueryResult iiresult = CharacterDatabase.Query(stmt); if (iiresult) @@ -524,9 +524,21 @@ void bot_ai::CheckOwnerExpiry() Field* fields2 = iiresult->Fetch(); uint32 itemGuidLow = fields2[11].Get(); uint32 itemId = fields2[12].Get(); - Item* item = new Item; - ASSERT(item->LoadFromDB(itemGuidLow, ownerGuid, fields2, itemId)); - items.push_back(item); + uint8 item_idx = std::numeric_limits::max(); + + for (uint8 i = 0; i != BOT_INVENTORY_SIZE; ++i) + { + if (_equips[i] && _equips[i]->GetEntry() == itemId) + { + item_idx = i; + break; + } + } + if (item_idx >= BOT_INVENTORY_SIZE) + LOG_ERROR("npcbots", "bot_ai::CheckOwnerExpiry(): item id {} guid {} not found in bot's inventory!\n{}", itemId, itemGuidLow, me->GetGUID().ToString()); + + items.push_back(_equips[item_idx]); + _removeEquipment(item_idx); } while (iiresult->NextRow()); @@ -12375,6 +12387,28 @@ bool bot_ai::_canEquip(ItemTemplate const* newProto, uint8 slot, bool ignoreItem return false; } +void bot_ai::_removeEquipment(uint8 slot) +{ + Item* item = _equips[slot]; + if (!item) + return; //already unequipped + + _usableItemSlotsMask &= ~(1ul << slot); + + RemoveItemBonuses(slot); + ApplyItemSetBonuses(item, false); + + if (slot == BOT_SLOT_OFFHAND) + { + if (me->CanDualWield()) + me->SetCanDualWield(false); + if (!(me->GetCreatureTemplate()->flags_extra & CREATURE_FLAG_EXTRA_NO_BLOCK)) + const_cast(me->GetCreatureTemplate())->flags_extra |= CREATURE_FLAG_EXTRA_NO_BLOCK; + } + + _equips[slot] = nullptr; +} + bool bot_ai::_unequip(uint8 slot, ObjectGuid receiver) { EquipmentInfo const* einfo = BotDataMgr::GetBotEquipmentInfo(me->GetEntry()); @@ -12386,10 +12420,7 @@ bool bot_ai::_unequip(uint8 slot, ObjectGuid receiver) uint32 itemId = item->GetEntry(); - _usableItemSlotsMask &= ~(1ul << slot); - - RemoveItemBonuses(slot); - ApplyItemSetBonuses(item, false); + _removeEquipment(slot); //hand old weapon to master if (receiver && (slot > BOT_SLOT_RANGED || einfo->ItemEntry[slot] != itemId)) @@ -12451,16 +12482,6 @@ bool bot_ai::_unequip(uint8 slot, ObjectGuid receiver) } _updateEquips(slot, nullptr); - - //offhand check - if (slot == BOT_SLOT_OFFHAND) - { - if (me->CanDualWield()) - me->SetCanDualWield(false); - if (!(me->GetCreatureTemplate()->flags_extra & CREATURE_FLAG_EXTRA_NO_BLOCK)) - const_cast(me->GetCreatureTemplate())->flags_extra |= CREATURE_FLAG_EXTRA_NO_BLOCK; - } - return true; } diff --git a/src/server/game/AI/NpcBots/bot_ai.h b/src/server/game/AI/NpcBots/bot_ai.h index 38a21778ba8b09..6e1206e96dd713 100644 --- a/src/server/game/AI/NpcBots/bot_ai.h +++ b/src/server/game/AI/NpcBots/bot_ai.h @@ -623,6 +623,7 @@ class bot_ai : public CreatureAI bool _canUseRanged() const; bool _canUseRelic() const; bool _canEquip(ItemTemplate const* newProto, uint8 slot, bool ignoreItemLevel, Item const* newItem = nullptr) const; + void _removeEquipment(uint8 slot); bool _unequip(uint8 slot, ObjectGuid receiver); bool _equip(uint8 slot, Item* newItem, ObjectGuid receiver); bool _resetEquipment(uint8 slot, ObjectGuid receiver);