/* * This file is part of the TrinityCore 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, see . */ #ifndef _SPELLINFO_H #define _SPELLINFO_H #include "SharedDefines.h" #include "Util.h" #include "DBCStructure.h" #include "Object.h" #include "SpellAuraDefines.h" class AuraEffect; class Item; class Player; class Spell; class SpellMgr; class SpellInfo; class Unit; class WorldObject; struct Condition; struct SpellChainNode; struct SpellModifier; enum WeaponAttackType : uint8; enum SpellTargetSelectionCategories { TARGET_SELECT_CATEGORY_NYI, TARGET_SELECT_CATEGORY_DEFAULT, TARGET_SELECT_CATEGORY_CHANNEL, TARGET_SELECT_CATEGORY_NEARBY, TARGET_SELECT_CATEGORY_CONE, TARGET_SELECT_CATEGORY_AREA, TARGET_SELECT_CATEGORY_TRAJ }; enum SpellTargetReferenceTypes { TARGET_REFERENCE_TYPE_NONE, TARGET_REFERENCE_TYPE_CASTER, TARGET_REFERENCE_TYPE_TARGET, TARGET_REFERENCE_TYPE_LAST, TARGET_REFERENCE_TYPE_SRC, TARGET_REFERENCE_TYPE_DEST }; enum SpellTargetObjectTypes : uint8 { TARGET_OBJECT_TYPE_NONE = 0, TARGET_OBJECT_TYPE_SRC, TARGET_OBJECT_TYPE_DEST, TARGET_OBJECT_TYPE_UNIT, TARGET_OBJECT_TYPE_UNIT_AND_DEST, TARGET_OBJECT_TYPE_GOBJ, TARGET_OBJECT_TYPE_GOBJ_ITEM, TARGET_OBJECT_TYPE_ITEM, TARGET_OBJECT_TYPE_CORPSE, // only for effect target type TARGET_OBJECT_TYPE_CORPSE_ENEMY, TARGET_OBJECT_TYPE_CORPSE_ALLY }; enum SpellTargetCheckTypes : uint8 { TARGET_CHECK_DEFAULT, TARGET_CHECK_ENTRY, TARGET_CHECK_ENEMY, TARGET_CHECK_ALLY, TARGET_CHECK_PARTY, TARGET_CHECK_RAID, TARGET_CHECK_RAID_CLASS, TARGET_CHECK_PASSENGER }; enum SpellTargetDirectionTypes { TARGET_DIR_NONE, TARGET_DIR_FRONT, TARGET_DIR_BACK, TARGET_DIR_RIGHT, TARGET_DIR_LEFT, TARGET_DIR_FRONT_RIGHT, TARGET_DIR_BACK_RIGHT, TARGET_DIR_BACK_LEFT, TARGET_DIR_FRONT_LEFT, TARGET_DIR_RANDOM, TARGET_DIR_ENTRY }; enum SpellEffectImplicitTargetTypes { EFFECT_IMPLICIT_TARGET_NONE = 0, EFFECT_IMPLICIT_TARGET_EXPLICIT, EFFECT_IMPLICIT_TARGET_CASTER }; // Spell clasification enum SpellSpecificType { SPELL_SPECIFIC_NORMAL = 0, SPELL_SPECIFIC_SEAL = 1, SPELL_SPECIFIC_AURA = 3, SPELL_SPECIFIC_STING = 4, SPELL_SPECIFIC_CURSE = 5, SPELL_SPECIFIC_ASPECT = 6, SPELL_SPECIFIC_TRACKER = 7, SPELL_SPECIFIC_WARLOCK_ARMOR = 8, SPELL_SPECIFIC_MAGE_ARMOR = 9, SPELL_SPECIFIC_ELEMENTAL_SHIELD = 10, SPELL_SPECIFIC_MAGE_POLYMORPH = 11, SPELL_SPECIFIC_JUDGEMENT = 13, SPELL_SPECIFIC_WARLOCK_CORRUPTION = 17, SPELL_SPECIFIC_FOOD = 19, SPELL_SPECIFIC_DRINK = 20, SPELL_SPECIFIC_FOOD_AND_DRINK = 21, SPELL_SPECIFIC_PRESENCE = 22, SPELL_SPECIFIC_CHARM = 23, SPELL_SPECIFIC_SCROLL = 24, SPELL_SPECIFIC_MAGE_ARCANE_BRILLANCE = 25, SPELL_SPECIFIC_WARRIOR_ENRAGE = 26, SPELL_SPECIFIC_PRIEST_DIVINE_SPIRIT = 27, SPELL_SPECIFIC_HAND = 28 }; enum SpellCustomAttributes { SPELL_ATTR0_CU_ENCHANT_PROC = 0x00000001, SPELL_ATTR0_CU_CONE_BACK = 0x00000002, SPELL_ATTR0_CU_CONE_LINE = 0x00000004, SPELL_ATTR0_CU_SHARE_DAMAGE = 0x00000008, SPELL_ATTR0_CU_NO_INITIAL_THREAT = 0x00000010, SPELL_ATTR0_CU_AURA_CC = 0x00000020, SPELL_ATTR0_CU_DONT_BREAK_STEALTH = 0x00000040, SPELL_ATTR0_CU_CAN_CRIT = 0x00000080, SPELL_ATTR0_CU_DIRECT_DAMAGE = 0x00000100, SPELL_ATTR0_CU_CHARGE = 0x00000200, SPELL_ATTR0_CU_PICKPOCKET = 0x00000400, SPELL_ATTR0_CU_ROLLING_PERIODIC = 0x00000800, SPELL_ATTR0_CU_NEGATIVE_EFF0 = 0x00001000, SPELL_ATTR0_CU_NEGATIVE_EFF1 = 0x00002000, SPELL_ATTR0_CU_NEGATIVE_EFF2 = 0x00004000, SPELL_ATTR0_CU_IGNORE_ARMOR = 0x00008000, SPELL_ATTR0_CU_REQ_TARGET_FACING_CASTER = 0x00010000, SPELL_ATTR0_CU_REQ_CASTER_BEHIND_TARGET = 0x00020000, SPELL_ATTR0_CU_ALLOW_INFLIGHT_TARGET = 0x00040000, SPELL_ATTR0_CU_NEEDS_AMMO_DATA = 0x00080000, SPELL_ATTR0_CU_BINARY_SPELL = 0x00100000, SPELL_ATTR0_CU_SCHOOLMASK_NORMAL_WITH_MAGIC = 0x00200000, SPELL_ATTR0_CU_DEPRECATED_LIQUID_AURA = 0x00400000, // DO NOT REUSE SPELL_ATTR0_CU_IS_TALENT = 0x00800000, // reserved for master branch SPELL_ATTR0_CU_AURA_CANNOT_BE_SAVED = 0x01000000, SPELL_ATTR0_CU_CAN_TARGET_ANY_PRIVATE_OBJECT = 0x02000000, // reserved for master branch SPELL_ATTR0_CU_NEGATIVE = SPELL_ATTR0_CU_NEGATIVE_EFF0 | SPELL_ATTR0_CU_NEGATIVE_EFF1 | SPELL_ATTR0_CU_NEGATIVE_EFF2 }; uint32 GetTargetFlagMask(SpellTargetObjectTypes objType); class TC_GAME_API SpellImplicitTargetInfo { private: Targets _target; public: SpellImplicitTargetInfo() : _target(Targets(0)) { } SpellImplicitTargetInfo(uint32 target); bool IsArea() const; SpellTargetSelectionCategories GetSelectionCategory() const; SpellTargetReferenceTypes GetReferenceType() const; SpellTargetObjectTypes GetObjectType() const; SpellTargetCheckTypes GetCheckType() const; SpellTargetDirectionTypes GetDirectionType() const; float CalcDirectionAngle() const; Targets GetTarget() const; uint32 GetExplicitTargetMask(bool& srcSet, bool& dstSet) const; private: struct StaticData { SpellTargetObjectTypes ObjectType; // type of object returned by target type SpellTargetReferenceTypes ReferenceType; // defines which object is used as a reference when selecting target SpellTargetSelectionCategories SelectionCategory; SpellTargetCheckTypes SelectionCheckType; // defines selection criteria SpellTargetDirectionTypes DirectionType; // direction for cone and dest targets }; static std::array _data; }; class TC_GAME_API SpellEffectInfo { friend class SpellInfo; SpellInfo const* _spellInfo; public: SpellEffIndex EffectIndex; SpellEffects Effect; AuraType ApplyAuraName; uint32 Amplitude; int32 DieSides; float RealPointsPerLevel; int32 BasePoints; float PointsPerComboPoint; float ValueMultiplier; float DamageMultiplier; float BonusMultiplier; int32 MiscValue; int32 MiscValueB; Mechanics Mechanic; SpellImplicitTargetInfo TargetA; SpellImplicitTargetInfo TargetB; SpellRadiusEntry const* RadiusEntry; uint32 ChainTarget; uint32 ItemType; uint32 TriggerSpell; flag96 SpellClassMask; std::vector* ImplicitTargetConditions; SpellEffectInfo(); explicit SpellEffectInfo(SpellEntry const* spellEntry, SpellInfo const* spellInfo, uint8 effIndex); SpellEffectInfo(SpellEffectInfo const&) = delete; SpellEffectInfo(SpellEffectInfo&&) noexcept; SpellEffectInfo& operator=(SpellEffectInfo const&) = delete; SpellEffectInfo& operator=(SpellEffectInfo&&) noexcept; ~SpellEffectInfo(); bool IsEffect() const; bool IsEffect(SpellEffects effectName) const; bool IsAura() const; bool IsAura(AuraType aura) const; bool IsTargetingArea() const; bool IsAreaAuraEffect() const; bool IsUnitOwnedAuraEffect() const; int32 CalcValue(WorldObject const* caster = nullptr, int32 const* basePoints = nullptr) const; int32 CalcBaseValue(int32 value) const; float CalcValueMultiplier(WorldObject* caster, Spell* spell = nullptr) const; float CalcDamageMultiplier(WorldObject* caster, Spell* spell = nullptr) const; bool HasRadius() const; float CalcRadius(WorldObject* caster = nullptr, Spell* = nullptr) const; uint32 GetProvidedTargetMask() const; uint32 GetMissingTargetMask(bool srcSet = false, bool destSet = false, uint32 mask = 0) const; SpellEffectImplicitTargetTypes GetImplicitTargetType() const; SpellTargetObjectTypes GetUsedTargetObjectType() const; struct ImmunityInfo; ImmunityInfo const* GetImmunityInfo() const { return _immunityInfo.get(); } private: struct StaticData { SpellEffectImplicitTargetTypes ImplicitTargetType; // defines what target can be added to effect target list if there's no valid target type provided for effect SpellTargetObjectTypes UsedTargetObjectType; // defines valid target object type for spell effect }; static std::array _data; std::unique_ptr _immunityInfo; }; struct TC_GAME_API SpellDiminishInfo { DiminishingGroup DiminishGroup = DIMINISHING_NONE; DiminishingReturnsType DiminishReturnType = DRTYPE_NONE; DiminishingLevels DiminishMaxLevel = DIMINISHING_LEVEL_IMMUNE; int32 DiminishDurationLimit = 0; }; class TC_GAME_API SpellInfo { friend class SpellMgr; public: uint32 Id; SpellCategoryEntry const* CategoryEntry; uint32 Dispel; uint32 Mechanic; uint32 Attributes; uint32 AttributesEx; uint32 AttributesEx2; uint32 AttributesEx3; uint32 AttributesEx4; uint32 AttributesEx5; uint32 AttributesEx6; uint32 AttributesEx7; uint32 AttributesCu; uint64 Stances; uint64 StancesNot; uint32 Targets; uint32 TargetCreatureType; uint32 RequiresSpellFocus; uint32 FacingCasterFlags; uint32 CasterAuraState; uint32 TargetAuraState; uint32 CasterAuraStateNot; uint32 TargetAuraStateNot; uint32 CasterAuraSpell; uint32 TargetAuraSpell; uint32 ExcludeCasterAuraSpell; uint32 ExcludeTargetAuraSpell; SpellCastTimesEntry const* CastTimeEntry; uint32 RecoveryTime; uint32 CategoryRecoveryTime; uint32 StartRecoveryCategory; uint32 StartRecoveryTime; uint32 InterruptFlags; uint32 AuraInterruptFlags; uint32 ChannelInterruptFlags; uint32 ProcFlags; uint32 ProcChance; uint32 ProcCharges; uint32 MaxLevel; uint32 BaseLevel; uint32 SpellLevel; SpellDurationEntry const* DurationEntry; Powers PowerType; uint32 ManaCost; uint32 ManaCostPerlevel; uint32 ManaPerSecond; uint32 ManaPerSecondPerLevel; uint32 ManaCostPercentage; uint32 RuneCostID; SpellRangeEntry const* RangeEntry; float Speed; uint32 StackAmount; std::array Totem; std::array Reagent; std::array ReagentCount; int32 EquippedItemClass; int32 EquippedItemSubClassMask; int32 EquippedItemInventoryTypeMask; std::array TotemCategory; std::array SpellVisual; uint32 SpellIconID; uint32 ActiveIconID; uint32 Priority; std::array SpellName; std::array Rank; uint32 MaxTargetLevel; uint32 MaxAffectedTargets; uint32 SpellFamilyName; flag96 SpellFamilyFlags; uint32 DmgClass; uint32 PreventionType; int32 AreaGroupId; uint32 SchoolMask; std::array _effects; uint32 ExplicitTargetMask; SpellChainNode const* ChainEntry; SpellInfo(SpellEntry const* spellEntry); ~SpellInfo(); uint32 GetCategory() const; bool HasEffect(SpellEffects effect) const; bool HasAura(AuraType aura) const; bool HasAreaAuraEffect() const; bool HasOnlyDamageEffects() const; inline bool HasAttribute(SpellAttr0 attribute) const { return !!(Attributes & attribute); } inline bool HasAttribute(SpellAttr1 attribute) const { return !!(AttributesEx & attribute); } inline bool HasAttribute(SpellAttr2 attribute) const { return !!(AttributesEx2 & attribute); } inline bool HasAttribute(SpellAttr3 attribute) const { return !!(AttributesEx3 & attribute); } inline bool HasAttribute(SpellAttr4 attribute) const { return !!(AttributesEx4 & attribute); } inline bool HasAttribute(SpellAttr5 attribute) const { return !!(AttributesEx5 & attribute); } inline bool HasAttribute(SpellAttr6 attribute) const { return !!(AttributesEx6 & attribute); } inline bool HasAttribute(SpellAttr7 attribute) const { return !!(AttributesEx7 & attribute); } inline bool HasAttribute(SpellCustomAttributes customAttribute) const { return !!(AttributesCu & customAttribute); } bool IsExplicitDiscovery() const; bool IsLootCrafting() const; bool IsProfessionOrRiding() const; bool IsProfession() const; bool IsPrimaryProfession() const; bool IsPrimaryProfessionFirstRank() const; bool IsAbilityLearnedWithProfession() const; bool IsAbilityOfSkillType(uint32 skillType) const; bool IsAffectingArea() const; bool IsTargetingArea() const; bool NeedsExplicitUnitTarget() const; bool NeedsToBeTriggeredByCaster(SpellInfo const* triggeringSpell) const; bool IsSelfCast() const; bool IsPassive() const; bool IsAutocastable() const; bool IsStackableWithRanks() const; bool IsPassiveStackableWithRanks() const; bool IsMultiSlotAura() const; bool IsStackableOnOneSlotWithDifferentCasters() const; bool IsCooldownStartedOnEvent() const; bool IsDeathPersistent() const; bool IsRequiringDeadTarget() const; bool IsAllowingDeadTarget() const; bool IsGroupBuff() const; bool CanBeUsedInCombat() const; bool IsPositive() const; bool IsPositiveEffect(uint8 effIndex) const; bool IsChanneled() const; bool IsMoveAllowedChannel() const; bool NeedsComboPoints() const; bool IsNextMeleeSwingSpell() const; bool IsBreakingStealth() const; bool IsRangedWeaponSpell() const; bool IsAutoRepeatRangedSpell() const; bool HasInitialAggro() const; WeaponAttackType GetAttackType() const; bool IsItemFitToSpellRequirements(Item const* item) const; bool IsAffected(uint32 familyName, flag96 const& familyFlags) const; bool IsAffectedBySpellMods() const; bool IsAffectedBySpellMod(SpellModifier const* mod) const; bool CanPierceImmuneAura(SpellInfo const* auraSpellInfo) const; bool CanDispelAura(SpellInfo const* auraSpellInfo) const; bool IsSingleTarget() const; bool IsAuraExclusiveBySpecificWith(SpellInfo const* spellInfo) const; bool IsAuraExclusiveBySpecificPerCasterWith(SpellInfo const* spellInfo) const; SpellCastResult CheckShapeshift(uint32 form) const; SpellCastResult CheckLocation(uint32 map_id, uint32 zone_id, uint32 area_id, Player const* player = nullptr, bool strict = true) const; SpellCastResult CheckTarget(WorldObject const* caster, WorldObject const* target, bool implicit = true) const; SpellCastResult CheckExplicitTarget(WorldObject const* caster, WorldObject const* target, Item const* itemTarget = nullptr) const; SpellCastResult CheckVehicle(Unit const* caster) const; bool CheckTargetCreatureType(Unit const* target) const; SpellSchoolMask GetSchoolMask() const; uint32 GetAllEffectsMechanicMask() const; uint32 GetEffectMechanicMask(SpellEffIndex effIndex) const; uint32 GetSpellMechanicMaskByEffectMask(uint32 effectMask) const; Mechanics GetEffectMechanic(SpellEffIndex effIndex) const; uint32 GetDispelMask() const; static uint32 GetDispelMask(DispelType type); uint32 GetExplicitTargetMask() const; AuraStateType GetAuraState() const; SpellSpecificType GetSpellSpecific() const; float GetMinRange(bool positive = false) const; float GetMaxRange(bool positive = false, WorldObject* caster = nullptr, Spell* spell = nullptr) const; int32 GetDuration() const; int32 GetMaxDuration() const; uint32 GetMaxTicks() const; uint32 CalcCastTime(Spell* spell = nullptr) const; uint32 GetRecoveryTime() const; int32 CalcPowerCost(WorldObject const* caster, SpellSchoolMask schoolMask, Spell* spell = nullptr) const; bool IsRanked() const; uint8 GetRank() const; SpellInfo const* GetFirstRankSpell() const; SpellInfo const* GetLastRankSpell() const; SpellInfo const* GetNextRankSpell() const; SpellInfo const* GetPrevRankSpell() const; SpellInfo const* GetAuraRankForLevel(uint8 level) const; bool IsRankOf(SpellInfo const* spellInfo) const; bool IsDifferentRankOf(SpellInfo const* spellInfo) const; bool IsHighRankOf(SpellInfo const* spellInfo) const; std::array const& GetEffects() const { return _effects; } SpellEffectInfo const& GetEffect(SpellEffIndex index) const { ASSERT(index < _effects.size()); return _effects[index]; } // spell diminishing returns DiminishingGroup GetDiminishingReturnsGroupForSpell(bool triggered) const; DiminishingReturnsType GetDiminishingReturnsGroupType(bool triggered) const; DiminishingLevels GetDiminishingReturnsMaxLevel(bool triggered) const; int32 GetDiminishingReturnsLimitDuration(bool triggered) const; // spell immunities void ApplyAllSpellImmunitiesTo(Unit* target, SpellEffectInfo const& spellEffectInfo, bool apply) const; bool CanSpellProvideImmunityAgainstAura(SpellInfo const* auraSpellInfo) const; bool SpellCancelsAuraEffect(AuraEffect const* aurEff) const; uint32 GetAllowedMechanicMask() const; uint32 GetMechanicImmunityMask(Unit* caster) const; private: // loading helpers void _InitializeExplicitTargetMask(); void _InitializeSpellPositivity(); void _LoadSpellSpecific(); void _LoadAuraState(); void _LoadSpellDiminishInfo(); void _LoadImmunityInfo(); std::array& _GetEffects() { return _effects; } SpellEffectInfo& _GetEffect(SpellEffIndex index) { ASSERT(index < _effects.size()); return _effects[index]; } // unloading helpers void _UnloadImplicitTargetConditionLists(); SpellSpecificType _spellSpecific; AuraStateType _auraState; SpellDiminishInfo _diminishInfoNonTriggered; SpellDiminishInfo _diminishInfoTriggered; uint32 _allowedMechanicMask; }; #endif // _SPELLINFO_H