From 8ae2cca85af8297a14002577390d665713f6f938 Mon Sep 17 00:00:00 2001 From: schell244 Date: Fri, 20 Dec 2024 07:25:40 +0100 Subject: [PATCH] Rewrite Golemagg and fix 'Golemagg's Trust' (#2835) --- sql/migrations/20241117202208_world.sql | 24 ++ src/game/Spells/SpellAuras.cpp | 45 ++- .../molten_core/boss_golemagg.cpp | 363 ++++++++++++------ 3 files changed, 319 insertions(+), 113 deletions(-) create mode 100644 sql/migrations/20241117202208_world.sql diff --git a/sql/migrations/20241117202208_world.sql b/sql/migrations/20241117202208_world.sql new file mode 100644 index 00000000000..d72258bacd1 --- /dev/null +++ b/sql/migrations/20241117202208_world.sql @@ -0,0 +1,24 @@ +DROP PROCEDURE IF EXISTS add_migration; +DELIMITER ?? +CREATE PROCEDURE `add_migration`() +BEGIN +DECLARE v INT DEFAULT 1; +SET v = (SELECT COUNT(*) FROM `migrations` WHERE `id`='20241117202208'); +IF v = 0 THEN +INSERT INTO `migrations` VALUES ('20241117202208'); +-- Add your query below. + +-- Golemagg - add missing auras +-- 13879 Magma Splash - Golemagg splashes magma at his attackers, dealing Fire damage over time and reducing their armor. This ability can stack. +-- 20556 Golemagg's Trust - Golemagg's Core Ragers will deal increased damage and have 50% increased attack speed if tanked too close to Golemagg. +-- 18943 Double Attack - Proc chance: 50%, 3s cooldown +UPDATE `creature_template` SET `auras` = "13879 20556 18943" WHERE `entry` = 11988; + +DELETE FROM `script_texts` WHERE `entry` = -1409002; -- %s refuses to die while its master is in trouble. (2) -> 7865 + +-- End of migration. +END IF; +END?? +DELIMITER ; +CALL add_migration(); +DROP PROCEDURE IF EXISTS add_migration; diff --git a/src/game/Spells/SpellAuras.cpp b/src/game/Spells/SpellAuras.cpp index 40dd0b5cd67..7417d796cfa 100644 --- a/src/game/Spells/SpellAuras.cpp +++ b/src/game/Spells/SpellAuras.cpp @@ -1558,7 +1558,7 @@ void Aura::TriggerSpell() for (auto const& it : pList) { Player* pPlayer = it.getSource(); - if (pPlayer->GetGUID() == casterGUID) continue; + if (pPlayer->GetGUID() == casterGUID.GetRawValue()) continue; if (!pPlayer) continue; if (pPlayer->IsDead()) continue; // 2d distance should be good enough @@ -1826,6 +1826,12 @@ void Aura::HandleAuraDummy(bool apply, bool Real) } return; } + case 20556: // Golemagg's Trust + { + m_isPeriodic = true; + m_modifier.periodictime = 1000; + return; + } case 22646: // Goblin Rocket Helmet { if (Unit* caster = GetCaster()) @@ -2062,8 +2068,18 @@ void Aura::HandleAuraDummy(bool apply, bool Real) case 23183: // Mark of Frost { if (m_removeMode == AURA_REMOVE_BY_DEATH) - target->CastSpell(target, 23182, true, nullptr, this); - return; + { + if (Unit* caster = GetCaster()) + { + // Azuregos + // TODO verify: Only cast Mark of Frost on targets nearby when engaged? + // if (caster->IsInCombat()) + // { + target->CastSpell(target, 23182, true, nullptr, this); + // } + } + } + return; } case 28169: // Mutating Injection { @@ -6790,6 +6806,29 @@ void Aura::PeriodicDummyTick() return; } + case 20556: // Golemagg's Trust + { + if (Unit* pCaster = GetCaster()) + { + if (pCaster->IsDead() && !pCaster->IsInCombat()) + { + return; + } + // Golemagg's Core Ragers will deal increased damage + // and have 50% increased attack speed if tanked too close to Golemagg. + std::list addList; + pCaster->GetCreatureListWithEntryInGrid(addList, 11672, 30.0f); + if (!addList.empty()) + { + for (const auto& itr : addList) + { + // Golemagg's Trust Buff + pCaster->CastSpell(itr, 20553, true, nullptr, this); + } + } + } + return; + } } break; } diff --git a/src/scripts/eastern_kingdoms/burning_steppes/molten_core/boss_golemagg.cpp b/src/scripts/eastern_kingdoms/burning_steppes/molten_core/boss_golemagg.cpp index e13aa5ee7ee..b1287a99f68 100644 --- a/src/scripts/eastern_kingdoms/burning_steppes/molten_core/boss_golemagg.cpp +++ b/src/scripts/eastern_kingdoms/burning_steppes/molten_core/boss_golemagg.cpp @@ -14,169 +14,272 @@ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ -/* ScriptData -SDName: Boss_Golemagg -SD%Complete: 75 -SDComment: Timers need to be confirmed, Golemagg's Trust need to be checked -SDCategory: Molten Core -EndScriptData */ - -/* - Golemagg and his adds should evade if dogs are pulled too far away from Golemagg -*/ - #include "scriptPCH.h" #include "molten_core.h" +#include "Utilities/EventMap.h" -enum +enum Golemagg : uint32 { - SPELL_MAGMASPLASH = 13879, - SPELL_PYROBLAST = 20228, - SPELL_EARTHQUAKE = 19798, - SPELL_ENRAGE = 19953, - SPELL_GOLEMAGG_TRUST = 20553, - - // Core Rager - EMOTE_LOWHP = -1409002, - SPELL_MANGLE = 19820, - SPELL_TRASH = 3391 + // Auras (set in db - creature_template) + // SPELL_MAGMASPLASH = 13879, + // SPELL_GOLEMAGG_TRUST_AURA = 20556, + // SPELL_DOUBLE_ATTACK = 18943, + + // Spells + SPELL_PYROBLAST = 20228, + SPELL_EARTHQUAKE = 19798, + SPELL_GOLEMAGG_TRUST = 20553, + SPELL_ATTRACK_RAGER = 20544, + + // Events + EVENT_PYROBLAST = 1, + EVENT_EARTHQUAKE = 2, + EVENT_GOLEMAGG_TRUST = 3 }; struct boss_golemaggAI : public ScriptedAI { - boss_golemaggAI(Creature* pCreature) : ScriptedAI(pCreature) + explicit boss_golemaggAI(Creature* pCreature) : ScriptedAI(pCreature) { - m_pInstance = (ScriptedInstance*)pCreature->GetInstanceData(); - Reset(); + m_pInstance = static_cast(pCreature->GetInstanceData()); + boss_golemaggAI::Reset(); } - ScriptedInstance* m_pInstance; + EventMap m_CombatEvents{}; + ScriptedInstance* m_pInstance{}; + std::list m_addList; - uint32 m_uiPyroblastTimer; - uint32 m_uiEarthquakeTimer; - uint32 TickTimer; - bool m_bEnraged; + bool m_bEnraged = false; void Reset() override { - m_uiPyroblastTimer = 7 * IN_MILLISECONDS; // These timers are probably wrong - m_uiEarthquakeTimer = 3 * IN_MILLISECONDS; - TickTimer = 10000; m_bEnraged = false; + m_CombatEvents.Reset(); + m_addList.clear(); if (m_pInstance && m_creature->IsAlive()) + { m_pInstance->SetData(TYPE_GOLEMAGG, NOT_STARTED); + } - - std::list ChiensListe; - GetCreatureListWithEntryInGrid(ChiensListe, m_creature, 11672, 150.0f); - if (!ChiensListe.empty()) + std::list addList; + GetCreatureListWithEntryInGrid(addList, m_creature, NPC_CORE_RAGER, 250.0f); + if (!addList.empty()) { - for (const auto& itr : ChiensListe) + for (const auto& itr : addList) { if (itr->GetDeathState() == ALIVE) + { itr->DealDamage(itr, itr->GetHealth(), nullptr, DIRECT_DAMAGE, SPELL_SCHOOL_MASK_NORMAL, nullptr, false); + } + itr->Respawn(); } } - // available in creature_addon - // m_creature->CastSpell(m_creature, SPELL_MAGMASPLASH, true); } - void Aggro(Unit* pWho) override + void Aggro(Unit*) override { if (m_pInstance) + { m_pInstance->SetData(TYPE_GOLEMAGG, IN_PROGRESS); + } + + // Store add guids + std::list adds; + GetCreatureListWithEntryInGrid(adds, m_creature, NPC_CORE_RAGER, 150.0f); + m_addList.clear(); + for (const auto& itr : adds) + { + m_addList.push_back(itr->GetObjectGuid()); + } + + ScheduleCombatEvents(); } - void JustDied(Unit* pKiller) override + void EnterEvadeMode() override + { + KillAdds(true); + ScriptedAI::EnterEvadeMode(); + } + + void JustDied(Unit*) override { if (m_pInstance) + { m_pInstance->SetData(TYPE_GOLEMAGG, DONE); + } + + KillAdds(false); } - void UpdateAI(uint32 const uiDiff) override + void KillAdds(bool respawn) { - if (!m_creature->SelectHostileTarget() || !m_creature->GetVictim()) - return; - - // Pyroblast - if (m_uiPyroblastTimer < uiDiff) + if (!m_addList.empty()) { - if (Unit* pTarget = m_creature->SelectAttackingTarget(ATTACKING_TARGET_RANDOM, 0)) + for (const auto& itr : m_addList) { - if (DoCastSpellIfCan(pTarget, SPELL_PYROBLAST) == CAST_OK) - m_uiPyroblastTimer = (3 + rand() % 4) * IN_MILLISECONDS; + if (auto coreRager = m_creature->GetMap()->GetCreature(itr)) + { + coreRager->DisappearAndDie(); + if (respawn) + { + coreRager->Respawn(); + } + } } } - else - m_uiPyroblastTimer -= uiDiff; + } - // Enrage - if (!m_bEnraged && m_creature->GetHealthPercent() < 10.0f) + void DamageTaken(Unit*, uint32& uiDamage) override + { + if (!m_bEnraged) { - if (DoCastSpellIfCan(m_creature, SPELL_ENRAGE) == CAST_OK) - m_bEnraged = true; + // At 10% health, Golemagg will Earthquake, + // dealing Physical damage to all players every couple seconds. + if (m_creature->GetHealthPercent() < 10.f) + { + if (DoCastSpellIfCan(m_creature, SPELL_ATTRACK_RAGER) == CAST_OK) + { + m_bEnraged = true; + m_CombatEvents.ScheduleEvent(EVENT_EARTHQUAKE, Seconds(5)); + } + } } + } - if (TickTimer < uiDiff) + void UpdateAI(uint32 const uiDiff) override + { + if (!m_creature->SelectHostileTarget() || !m_creature->GetVictim()) { - if (DoCastSpellIfCan(m_creature, SPELL_GOLEMAGG_TRUST) == CAST_OK) - TickTimer = 2000; + return; } - else TickTimer -= uiDiff; - // Earthquake - if (m_bEnraged) + m_CombatEvents.Update(uiDiff); + UpdateEvents(); + + DoMeleeAttackIfReady(); + } + + void ScheduleCombatEvents() + { + m_CombatEvents.ScheduleEvent(EVENT_PYROBLAST, Seconds(7)); + m_CombatEvents.ScheduleEvent(EVENT_GOLEMAGG_TRUST, Seconds(10)); + } + + void UpdateEvents() + { + while (const uint32 eventId = m_CombatEvents.ExecuteEvent()) { - if (m_uiEarthquakeTimer < uiDiff) + switch (eventId) { - if (DoCastSpellIfCan(m_creature->GetVictim(), SPELL_EARTHQUAKE) == CAST_OK) - m_uiEarthquakeTimer = 3 * IN_MILLISECONDS; + case EVENT_PYROBLAST: + { + // Golemagg Pyroblasts a random player, + // dealing Fire damage and leaving an additional Fire DOT on them. + if (Unit* pTarget = m_creature->SelectAttackingTarget(ATTACKING_TARGET_RANDOM, 0)) + { + if (DoCastSpellIfCan(pTarget, SPELL_PYROBLAST) == CAST_OK) + { + m_CombatEvents.Repeat(Seconds(7)); + return; + } + } + + // Cast Failed: Try again in 1s + m_CombatEvents.Repeat(Seconds(1)); + return; + } + case EVENT_EARTHQUAKE: + { + if (DoCastSpellIfCan(m_creature->GetVictim(), SPELL_EARTHQUAKE) == CAST_OK) + { + m_CombatEvents.Repeat(Seconds(5)); + return; + } + + // Cast Failed: Try again in 1s + m_CombatEvents.Repeat(Seconds(1)); + return; + } + case EVENT_GOLEMAGG_TRUST: + { + if (DoCastSpellIfCan(m_creature, SPELL_GOLEMAGG_TRUST) == CAST_OK) + { + m_CombatEvents.Repeat(Seconds(2)); + return; + } + // Cast Failed: Try again in 1s + m_CombatEvents.Repeat(Seconds(1)); + return; + } } - else - m_uiEarthquakeTimer -= uiDiff; } + } +}; +CreatureAI* GetAI_boss_golemagg(Creature* pCreature) +{ + return new boss_golemaggAI(pCreature); +} - DoMeleeAttackIfReady(); - } +enum CoreRager : uint32 +{ + // Spells + // SPELL_TRASH = 3391, // Custom? + SPELL_MANGLE = 19820, + SPELL_FULL_HEAL = 17683, + + // Texts + EMOTE_LOWHP = 7865, + + // Events + EVENT_MANGLE = 1, + EVENT_TRASH = 2, + EVENT_CHECK_LEASH = 3 }; struct mob_core_ragerAI : public ScriptedAI { - mob_core_ragerAI(Creature* pCreature) : ScriptedAI(pCreature) + explicit mob_core_ragerAI(Creature* pCreature) : ScriptedAI(pCreature) { - m_pInstance = (ScriptedInstance*)pCreature->GetInstanceData(); + m_pInstance = static_cast(pCreature->GetInstanceData()); Reset(); } - ScriptedInstance* m_pInstance; - uint32 m_uiMangleTimer; - uint32 TickTimer; + EventMap m_CombatEvents{}; + ScriptedInstance* m_pInstance{}; void Reset() override { - TickTimer = 1000; - m_uiMangleTimer = 7 * IN_MILLISECONDS; // These times are probably wrong + m_CombatEvents.Reset(); } - void DamageTaken(Unit* pDoneBy, uint32& uiDamage) override + void Aggro(Unit*) override { - if (m_pInstance) + ScheduleCombatEvents(); + } + + void DamageTaken(Unit*, uint32& uiDamage) override + { + if (!m_pInstance) { - if (Creature* pGolemagg = m_pInstance->instance->GetCreature(m_pInstance->GetData64(DATA_GOLEMAGG))) + return; + } + + if (m_pInstance->GetData(TYPE_GOLEMAGG) == DONE) + { + return; + } + + if (m_creature->HealthBelowPctDamaged(50, uiDamage)) + { + uiDamage = 0; + DoScriptText(EMOTE_LOWHP, m_creature); + SpellCastResult result = DoCastSpellIfCan(m_creature->GetVictim(), SPELL_FULL_HEAL); + if (result != CAST_OK) { - if (pGolemagg->IsAlive()) - { - if (m_creature->GetHealthPercent() < 50.0f) - { - DoScriptText(EMOTE_LOWHP, m_creature); - m_creature->SetHealth(m_creature->GetMaxHealth()); - } - } - else - m_creature->DisappearAndDie(); + m_creature->SetHealth(m_creature->GetMaxHealth()); } } } @@ -184,33 +287,73 @@ struct mob_core_ragerAI : public ScriptedAI void UpdateAI(uint32 const uiDiff) override { if (!m_creature->SelectHostileTarget() || !m_creature->GetVictim()) - return; - - // Mangle - if (m_uiMangleTimer < uiDiff) { - if (DoCastSpellIfCan(m_creature->GetVictim(), SPELL_MANGLE) == CAST_OK) - m_uiMangleTimer = 10 * IN_MILLISECONDS; + return; } - else - m_uiMangleTimer -= uiDiff; - if (TickTimer < uiDiff) - { - TickTimer = 1000; - if (!m_creature->HasAura(SPELL_TRASH) && !(bool)(rand() % 10)) - m_creature->CastSpell(m_creature, SPELL_TRASH, true); - } - else TickTimer -= uiDiff; + m_CombatEvents.Update(uiDiff); + UpdateEvents(); DoMeleeAttackIfReady(); } -}; -CreatureAI* GetAI_boss_golemagg(Creature* pCreature) -{ - return new boss_golemaggAI(pCreature); -} + void ScheduleCombatEvents() + { + // m_CombatEvents.ScheduleEvent(EVENT_TRASH, Seconds(1)); + m_CombatEvents.ScheduleEvent(EVENT_MANGLE, Seconds(7)); + m_CombatEvents.ScheduleEvent(EVENT_CHECK_LEASH, Seconds(3)); + } + + void UpdateEvents() + { + while (const uint32 eventId = m_CombatEvents.ExecuteEvent()) + { + switch (eventId) + { + /* + case EVENT_TRASH: + { + if (!m_creature->HasAura(SPELL_TRASH)) + { + m_creature->CastSpell(m_creature, SPELL_TRASH, true); + } + + m_CombatEvents.Repeat(Seconds(urand(1, 10))); + return; + } + */ + case EVENT_MANGLE: + { + if (DoCastSpellIfCan(m_creature->GetVictim(), SPELL_MANGLE) == CAST_OK) + { + m_CombatEvents.Repeat(Seconds(10)); + return; + } + + // Cast Failed: Try again in 1s + m_CombatEvents.Repeat(Seconds(1)); + return; + } + case EVENT_CHECK_LEASH: + { + if (Creature* pGolemagg = m_pInstance->instance->GetCreature(m_pInstance->GetData64(DATA_GOLEMAGG))) + { + if (m_creature->GetDistance2d(pGolemagg) > 100.f) + { + if (auto pGolemaggAI = dynamic_cast(pGolemagg->AI())) + { + pGolemaggAI->EnterEvadeMode(); + } + } + } + + m_CombatEvents.Repeat(Seconds(3)); + return; + } + } + } + } +}; CreatureAI* GetAI_mob_core_rager(Creature* pCreature) {