warrior

package
v0.0.180 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Jan 18, 2025 License: MIT Imports: 6 Imported by: 0

Documentation

Index

Constants

View Source
const (
	SpellFlagBleed = core.SpellFlagAgentReserved1
	ArmsTree       = 0
	FuryTree       = 1
	ProtTree       = 2
)
View Source
const (
	SpellMaskSpecialAttack int64 = 1 << iota

	// Abilities that don't cost rage and aren't attacks
	SpellMaskBattleShout
	SpellMaskBerserkerRage
	SpellMaskCommandingShout
	SpellMaskRecklessness
	SpellMaskShieldWall
	SpellMaskLastStand
	SpellMaskDeadlyCalm
	SpellMaskCharge

	// Abilities that cost rage but aren't attacks
	SpellMaskDemoShout
	SpellMaskInnerRage
	SpellMaskShieldBlock
	SpellMaskDeathWish
	SpellMaskSweepingStrikes

	// Special attacks
	SpellMaskCleave
	SpellMaskColossusSmash
	SpellMaskExecute
	SpellMaskHeroicStrike
	SpellMaskHeroicThrow
	SpellMaskOverpower
	SpellMaskRend
	SpellMaskRevenge
	SpellMaskShatteringThrow
	SpellMaskSlam
	SpellMaskSunderArmor
	SpellMaskThunderClap
	SpellMaskWhirlwind
	SpellMaskWhirlwindOh
	SpellMaskShieldSlam
	SpellMaskConcussionBlow
	SpellMaskDevastate
	SpellMaskShockwave
	SpellMaskVictoryRush
	SpellMaskBloodthirst
	SpellMaskRagingBlow
	SpellMaskMortalStrike
	SpellMaskBladestorm
	SpellMaskHeroicLeap

	SpellMaskShouts = SpellMaskCommandingShout | SpellMaskBattleShout
)
View Source
const EnableOverpowerTag = "EnableOverpower"
View Source
const EnrageTag = "EnrageEffect"
View Source
const InnerRageExclusionTag = "InnerRageDeadlyCalm"
View Source
const ShoutExpirationThreshold = time.Second * 3
View Source
const SpellMaskNone int64 = 0

Variables

View Source
var ItemSetColossalDragonplateArmor = core.NewItemSet(core.ItemSet{
	Name: "Colossal Dragonplate Armor",
	Bonuses: map[int32]core.ApplySetBonus{
		2: func(agent core.Agent, setBonusAura *core.Aura) {
			character := agent.(WarriorAgent).GetWarrior()
			actionID := core.ActionID{SpellID: 105909}
			duration := time.Second * 6

			shieldAmt := 0.0
			shieldAura := character.NewDamageAbsorptionAura("Shield of Fury"+character.Label, actionID, duration, func(unit *core.Unit) float64 {
				return shieldAmt
			})

			setBonusAura.AttachProcTrigger(core.ProcTrigger{
				Name:           "Shield of Fury Trigger" + character.Label,
				Callback:       core.CallbackOnSpellHitDealt,
				ClassSpellMask: SpellMaskRevenge,
				Outcome:        core.OutcomeLanded,
				ProcChance:     1,
				Handler: func(sim *core.Simulation, spell *core.Spell, result *core.SpellResult) {
					if result.Target != character.CurrentTarget {
						return
					}

					shieldAmt = result.Damage * 0.2
					if shieldAmt > 1 {
						shieldAura.Activate(sim)
					}
				},
			})

			setBonusAura.ApplyOnExpire(func(aura *core.Aura, sim *core.Simulation) {
				shieldAmt = 0
			})
		},
		4: func(agent core.Agent, setBonusAura *core.Aura) {

		},
	},
})
View Source
var ItemSetColossalDragonplateBattlegear = core.NewItemSet(core.ItemSet{
	Name: "Colossal Dragonplate Battlegear",
	Bonuses: map[int32]core.ApplySetBonus{
		2: func(agent core.Agent, setBonusAura *core.Aura) {
			character := agent.(WarriorAgent).GetWarrior()

			mod := character.AddDynamicMod(core.SpellModConfig{
				ClassMask:  SpellMaskHeroicStrike,
				Kind:       core.SpellMod_PowerCost_Flat,
				FloatValue: -10,
			})

			actionID := core.ActionID{SpellID: 105860}
			buffAura := character.RegisterAura(core.Aura{
				Label:    "Volatile Outrage",
				ActionID: actionID,
				Duration: 15 * time.Second,
				OnGain: func(aura *core.Aura, sim *core.Simulation) {
					mod.Activate()
				},
				OnExpire: func(aura *core.Aura, sim *core.Simulation) {
					mod.Deactivate()
				},
			})

			setBonusAura.AttachProcTrigger(core.ProcTrigger{
				Name:           "Volatile Outrage Trigger",
				ActionID:       actionID,
				Callback:       core.CallbackOnCastComplete,
				ClassSpellMask: SpellMaskInnerRage,
				Handler: func(sim *core.Simulation, spell *core.Spell, result *core.SpellResult) {
					buffAura.Activate(sim)
				},
			})
		},
		4: func(agent core.Agent, setBonusAura *core.Aura) {
			warrior := agent.(WarriorAgent).GetWarrior()

			actionID := core.ActionID{SpellID: 108126}
			procCS := warrior.RegisterSpell(core.SpellConfig{
				ActionID:    actionID,
				SpellSchool: core.SpellSchoolPhysical,
				Flags:       core.SpellFlagNoOnDamageDealt | core.SpellFlagNoOnCastComplete,
				Cast: core.CastConfig{
					DefaultCast: core.Cast{
						GCD: 0,
					},
				},
				ApplyEffects: func(sim *core.Simulation, target *core.Unit, spell *core.Spell) {
					warrior.ColossusSmashAuras.Get(target).Activate(sim)
				},
			})

			baseProcTriggerConfig := func(config core.ProcTrigger) core.ProcTrigger {
				return core.ProcTrigger{
					Name:           config.Name,
					ClassSpellMask: config.ClassSpellMask,
					ProcChance:     config.ProcChance,
					ActionID:       actionID,
					Callback:       core.CallbackOnSpellHitDealt,
					Handler: func(sim *core.Simulation, spell *core.Spell, result *core.SpellResult) {
						procCS.Cast(sim, result.Target)
					},
				}
			}

			setBonusAura.MakeDependentProcTriggerAura(&warrior.Unit, baseProcTriggerConfig(core.ProcTrigger{
				Name:           "Warrior T13 4P Bloodthirst Trigger",
				ClassSpellMask: SpellMaskBloodthirst,
				ProcChance:     0.06,
			}))

			setBonusAura.MakeDependentProcTriggerAura(&warrior.Unit, baseProcTriggerConfig(core.ProcTrigger{
				Name:           "Warrior T13 4P Mortal Strike Trigger",
				ClassSpellMask: SpellMaskMortalStrike,
				ProcChance:     0.13,
			}))
		},
	},
})
View Source
var ItemSetEarthenBattleplate = core.NewItemSet(core.ItemSet{
	Name: "Earthen Battleplate",
	Bonuses: map[int32]core.ApplySetBonus{
		2: func(_ core.Agent, setBonusAura *core.Aura) {
			setBonusAura.AttachSpellMod(core.SpellModConfig{
				ClassMask:  SpellMaskShieldSlam,
				Kind:       core.SpellMod_DamageDone_Flat,
				FloatValue: 0.05,
			})
		},
		4: func(_ core.Agent, setBonusAura *core.Aura) {
			setBonusAura.AttachSpellMod(core.SpellModConfig{
				ClassMask:  SpellMaskShieldWall,
				Kind:       core.SpellMod_Cooldown_Multiplier,
				FloatValue: -0.5,
			})
		},
	},
})
View Source
var ItemSetEarthenWarplate = core.NewItemSet(core.ItemSet{
	Name: "Earthen Warplate",
	Bonuses: map[int32]core.ApplySetBonus{
		2: func(_ core.Agent, setBonusAura *core.Aura) {
			setBonusAura.AttachSpellMod(core.SpellModConfig{
				ClassMask:  SpellMaskBloodthirst | SpellMaskMortalStrike,
				Kind:       core.SpellMod_DamageDone_Flat,
				FloatValue: 0.05,
			})
		},
		4: func(agent core.Agent, setBonusAura *core.Aura) {
			character := agent.(WarriorAgent).GetWarrior()
			actionID := core.ActionID{SpellID: 90294}

			apDep := make([]*stats.StatDependency, 3)
			for i := 1; i <= 3; i++ {
				apDep[i-1] = character.NewDynamicMultiplyStat(stats.AttackPower, 1.0+float64(i)*0.01)
			}

			buff := character.RegisterAura(core.Aura{
				Label:     "Rage of the Ages",
				ActionID:  actionID,
				Duration:  30 * time.Second,
				MaxStacks: 3,
				OnStacksChange: func(aura *core.Aura, sim *core.Simulation, oldStacks, newStacks int32) {

					if oldStacks > 0 {
						character.DisableDynamicStatDep(sim, apDep[oldStacks-1])
					}
					if newStacks > 0 {
						character.EnableDynamicStatDep(sim, apDep[newStacks-1])
					}
				},
			})

			setBonusAura.AttachProcTrigger(core.ProcTrigger{
				Name:           "Rage of the Ages Trigger",
				ActionID:       actionID,
				Callback:       core.CallbackOnCastComplete,
				ClassSpellMask: SpellMaskOverpower | SpellMaskRagingBlow,
				Handler: func(sim *core.Simulation, spell *core.Spell, result *core.SpellResult) {
					buff.Activate(sim)
					buff.AddStack(sim)
				},
			})
		},
	},
})
View Source
var ItemSetGladiatorsBattlegear = core.NewItemSet(core.ItemSet{
	ID:   909,
	Name: "Gladiator's Battlegear",
	Bonuses: map[int32]core.ApplySetBonus{
		2: func(_ core.Agent, setBonusAura *core.Aura) {
			setBonusAura.AttachStatBuff(stats.Strength, 70)
		},
		4: func(_ core.Agent, setBonusAura *core.Aura) {
			setBonusAura.AttachStatBuff(stats.Strength, 90)
		},
	},
})
View Source
var ItemSetMoltenGiantBattleplate = core.NewItemSet(core.ItemSet{
	Name: "Molten Giant Battleplate",
	Bonuses: map[int32]core.ApplySetBonus{
		2: func(agent core.Agent, setBonusAura *core.Aura) {
			character := agent.(WarriorAgent).GetWarrior()

			cata.RegisterIgniteEffect(&character.Unit, cata.IgniteConfig{
				ActionID:           core.ActionID{SpellID: 23922}.WithTag(3),
				DisableCastMetrics: true,
				DotAuraLabel:       "Combust",
				IncludeAuraDelay:   true,
				SetBonusAura:       setBonusAura,

				ProcTrigger: core.ProcTrigger{
					Name:           "Combust",
					Callback:       core.CallbackOnSpellHitDealt,
					ClassSpellMask: SpellMaskShieldSlam,
					Outcome:        core.OutcomeLanded,
				},

				DamageCalculator: func(result *core.SpellResult) float64 {
					return result.Damage * 0.2
				},
			})
		},
		4: func(agent core.Agent, setBonusAura *core.Aura) {
			character := agent.(WarriorAgent).GetWarrior()

			aura := character.RegisterAura(core.Aura{
				Label:    "Item - Warrior T12 Protection 4P Bonus",
				ActionID: core.ActionID{SpellID: 99242},
				Duration: 10 * time.Second,
				OnGain: func(aura *core.Aura, sim *core.Simulation) {
					character.PseudoStats.BaseParryChance += 0.06
				},
				OnExpire: func(aura *core.Aura, sim *core.Simulation) {
					character.PseudoStats.BaseParryChance -= 0.06
				},
			})

			character.OnSpellRegistered(func(spell *core.Spell) {
				if !spell.Matches(SpellMaskShieldBlock) {
					return
				}

				character.ShieldBlockAura.ApplyOnExpire(func(_ *core.Aura, sim *core.Simulation) {
					if setBonusAura.IsActive() {
						aura.Activate(sim)
					}
				})
			})
		},
	},
})
View Source
var ItemSetMoltenGiantWarplate = core.NewItemSet(core.ItemSet{
	Name: "Molten Giant Warplate",
	Bonuses: map[int32]core.ApplySetBonus{
		2: func(agent core.Agent, setBonusAura *core.Aura) {
			character := agent.(WarriorAgent).GetWarrior()
			actionID := core.ActionID{SpellID: 99233}

			talentReduction := time.Duration(character.Talents.BoomingVoice*3) * time.Second

			buff := character.RegisterAura(core.Aura{
				Label:    "Burning Rage",
				ActionID: actionID,
				Duration: 12*time.Second - talentReduction,
				OnGain: func(aura *core.Aura, sim *core.Simulation) {
					character.PseudoStats.SchoolDamageDealtMultiplier[stats.SchoolIndexPhysical] *= 1.1
				},
				OnExpire: func(aura *core.Aura, sim *core.Simulation) {
					character.PseudoStats.SchoolDamageDealtMultiplier[stats.SchoolIndexPhysical] /= 1.1
				},
			})

			setBonusAura.AttachProcTrigger(core.ProcTrigger{
				Name:           "Burning Rage Trigger",
				ActionID:       actionID,
				ClassSpellMask: SpellMaskShouts,
				Callback:       core.CallbackOnCastComplete,
				Handler: func(sim *core.Simulation, spell *core.Spell, result *core.SpellResult) {
					buff.Activate(sim)
				},
			})
		},
		4: func(agent core.Agent, setBonusAura *core.Aura) {
			character := agent.(WarriorAgent).GetWarrior()

			actionID := core.ActionID{SpellID: 99237}

			fieryAttackActionID := core.ActionID{}
			switch character.Spec {
			case proto.Spec_SpecArmsWarrior:
				fieryAttackActionID.SpellID = 12294
			case proto.Spec_SpecFuryWarrior:
				fieryAttackActionID.SpellID = 85288
			}

			fieryAttack := character.RegisterSpell(core.SpellConfig{
				ActionID:    fieryAttackActionID.WithTag(3),
				SpellSchool: core.SpellSchoolFire,
				ProcMask:    core.ProcMaskEmpty,
				Flags:       core.SpellFlagMeleeMetrics | core.SpellFlagPassiveSpell,

				CritMultiplier:   character.DefaultMeleeCritMultiplier(),
				DamageMultiplier: 1,
				ThreatMultiplier: 1,

				ApplyEffects: func(sim *core.Simulation, target *core.Unit, spell *core.Spell) {
					baseDamage := spell.Unit.MHNormalizedWeaponDamage(sim, spell.MeleeAttackPower())
					spell.CalcAndDealDamage(sim, target, baseDamage, spell.OutcomeMeleeWeaponSpecialHitAndCrit)
				},
			})

			setBonusAura.AttachProcTrigger(core.ProcTrigger{
				Name:           "Fiery Attack Trigger",
				ActionID:       actionID,
				Callback:       core.CallbackOnSpellHitDealt,
				ClassSpellMask: SpellMaskMortalStrike | SpellMaskRagingBlow,
				ProcChance:     0.3,
				Outcome:        core.OutcomeLanded,
				Handler: func(sim *core.Simulation, spell *core.Spell, result *core.SpellResult) {
					fieryAttack.Cast(sim, result.Target)
				},
			})
		},
	},
})
View Source
var TalentTreeSizes = [3]int{20, 21, 20}

Functions

This section is empty.

Types

type RageMultiplierCB

type RageMultiplierCB func() float64

Rather than update a variable somewhere for one effect (Fury's Unshackled Fury) just take a callback to fetch its multiplier when needed

type SpellEffectWeaponDmgPctConfig

type SpellEffectWeaponDmgPctConfig struct {
	BaseWeapon_Pct    float64
	Coefficient       float64
	EffectPerLevel    float64
	BaseSpellLevel    int32
	MaxSpellLevel     int32
	ClassSpellScaling float64
}

func (*SpellEffectWeaponDmgPctConfig) CalcAddedSpellDamage

func (config *SpellEffectWeaponDmgPctConfig) CalcAddedSpellDamage() float64

func (*SpellEffectWeaponDmgPctConfig) CalcSpellDamagePct

func (config *SpellEffectWeaponDmgPctConfig) CalcSpellDamagePct() float64

type Stance

type Stance uint8
const (
	StanceNone          = 0
	BattleStance Stance = 1 << iota
	DefensiveStance
	BerserkerStance
)

type Warrior

type Warrior struct {
	core.Character

	ClassSpellScaling float64

	Talents *proto.WarriorTalents

	WarriorInputs

	// Current state
	Stance                 Stance
	EnrageEffectMultiplier float64
	CriticalBlockChance    []float64 // Can be gained as non-prot via certain talents and spells
	PrecisionKnown         bool

	BattleShout     *core.Spell
	CommandingShout *core.Spell
	BattleStance    *core.Spell
	DefensiveStance *core.Spell
	BerserkerStance *core.Spell

	BerserkerRage     *core.Spell
	ColossusSmash     *core.Spell
	DemoralizingShout *core.Spell
	Execute           *core.Spell
	Overpower         *core.Spell
	Rend              *core.Spell
	Revenge           *core.Spell
	ShieldBlock       *core.Spell
	Slam              *core.Spell
	SunderArmor       *core.Spell
	ThunderClap       *core.Spell
	Whirlwind         *core.Spell
	DeepWounds        *core.Spell
	Charge            *core.Spell
	ChargeAura        *core.Aura

	HeroicStrike *core.Spell
	Cleave       *core.Spell

	BattleStanceAura    *core.Aura
	DefensiveStanceAura *core.Aura
	BerserkerStanceAura *core.Aura

	BerserkerRageAura *core.Aura
	BloodsurgeAura    *core.Aura
	SuddenDeathAura   *core.Aura
	ShieldBlockAura   *core.Aura
	ThunderstruckAura *core.Aura
	InnerRageAura     *core.Aura

	DemoralizingShoutAuras core.AuraArray
	SunderArmorAuras       core.AuraArray
	ThunderClapAuras       core.AuraArray
	ColossusSmashAuras     core.AuraArray
	// contains filtered or unexported fields
}

func NewWarrior

func NewWarrior(character *core.Character, talents string, inputs WarriorInputs) *Warrior

func (*Warrior) AddPartyBuffs

func (warrior *Warrior) AddPartyBuffs(_ *proto.PartyBuffs)

func (*Warrior) AddRaidBuffs

func (warrior *Warrior) AddRaidBuffs(raidBuffs *proto.RaidBuffs)

func (*Warrior) ApplyCommonTalents

func (warrior *Warrior) ApplyCommonTalents()

Applies the effects of "common" talents: talents in the first two rows of each tree that any spec could theoretically take Because cata restricts you to 10 points in a different tree, anything more is inaccessible. The rest of the trees are handled in each spec's implementation

func (*Warrior) ApplyGlyphs

func (warrior *Warrior) ApplyGlyphs()

func (*Warrior) CanApplySunderAura

func (warrior *Warrior) CanApplySunderAura(target *core.Unit) bool

func (*Warrior) GetCharacter

func (warrior *Warrior) GetCharacter() *core.Character

func (*Warrior) GetCriticalBlockChance

func (warrior *Warrior) GetCriticalBlockChance() float64

func (*Warrior) GetTentacles added in v0.0.119

func (warrior *Warrior) GetTentacles() []*cata.TentacleOfTheOldOnesPet

func (*Warrior) HasMajorGlyph

func (warrior *Warrior) HasMajorGlyph(glyph proto.WarriorMajorGlyph) bool

func (*Warrior) HasMinorGlyph

func (warrior *Warrior) HasMinorGlyph(glyph proto.WarriorMinorGlyph) bool

func (*Warrior) HasPrimeGlyph

func (warrior *Warrior) HasPrimeGlyph(glyph proto.WarriorPrimeGlyph) bool

func (*Warrior) Initialize

func (warrior *Warrior) Initialize()

func (*Warrior) IntensifyRageCooldown

func (warrior *Warrior) IntensifyRageCooldown(baseCd time.Duration) time.Duration

func (*Warrior) MakeShoutSpellHelper

func (warrior *Warrior) MakeShoutSpellHelper(actionID core.ActionID, spellMask int64, allyAuras core.AuraArray) *core.Spell

func (*Warrior) NewTentacleOfTheOldOnesPet added in v0.0.119

func (warrior *Warrior) NewTentacleOfTheOldOnesPet() *cata.TentacleOfTheOldOnesPet

func (*Warrior) RecklessnessDeadlyCalmLock

func (warrior *Warrior) RecklessnessDeadlyCalmLock() *core.Timer

Shared cooldown for Deadly Calm and Recklessness Activation

func (*Warrior) RegisterBerserkerRageSpell

func (warrior *Warrior) RegisterBerserkerRageSpell()

func (*Warrior) RegisterCharge

func (warrior *Warrior) RegisterCharge()

func (*Warrior) RegisterCleaveSpell

func (warrior *Warrior) RegisterCleaveSpell()

func (*Warrior) RegisterColossusSmash

func (warrior *Warrior) RegisterColossusSmash()

func (*Warrior) RegisterDeepWounds

func (warrior *Warrior) RegisterDeepWounds()

func (*Warrior) RegisterDemoralizingShoutSpell

func (warrior *Warrior) RegisterDemoralizingShoutSpell()

func (*Warrior) RegisterExecuteSpell

func (warrior *Warrior) RegisterExecuteSpell()

func (*Warrior) RegisterHeroicLeap

func (warrior *Warrior) RegisterHeroicLeap()

func (*Warrior) RegisterHeroicStrikeSpell

func (warrior *Warrior) RegisterHeroicStrikeSpell()

func (*Warrior) RegisterHeroicThrow

func (warrior *Warrior) RegisterHeroicThrow()

TODO: No patch notes for this ability, need to validate the damage and threat coefficients haven't changed

func (*Warrior) RegisterInnerRage

func (warrior *Warrior) RegisterInnerRage()

func (*Warrior) RegisterOverpowerSpell

func (warrior *Warrior) RegisterOverpowerSpell()

func (*Warrior) RegisterRecklessnessCD

func (warrior *Warrior) RegisterRecklessnessCD()

func (*Warrior) RegisterRendSpell

func (warrior *Warrior) RegisterRendSpell()

func (*Warrior) RegisterRevengeSpell

func (warrior *Warrior) RegisterRevengeSpell()

func (*Warrior) RegisterShatteringThrowCD

func (warrior *Warrior) RegisterShatteringThrowCD()

func (*Warrior) RegisterShieldBlockCD

func (warrior *Warrior) RegisterShieldBlockCD()

func (*Warrior) RegisterShieldWallCD

func (warrior *Warrior) RegisterShieldWallCD()

func (*Warrior) RegisterShouts

func (warrior *Warrior) RegisterShouts()

func (*Warrior) RegisterSlamSpell

func (warrior *Warrior) RegisterSlamSpell()

func (*Warrior) RegisterSunderArmor

func (warrior *Warrior) RegisterSunderArmor() *core.Spell

func (*Warrior) RegisterThunderClapSpell

func (warrior *Warrior) RegisterThunderClapSpell()

func (*Warrior) RegisterWhirlwindSpell

func (warrior *Warrior) RegisterWhirlwindSpell()

func (*Warrior) Reset

func (warrior *Warrior) Reset(_ *core.Simulation)

func (*Warrior) StanceMatches

func (warrior *Warrior) StanceMatches(other Stance) bool

func (*Warrior) TryApplySunderArmorEffect

func (warrior *Warrior) TryApplySunderArmorEffect(sim *core.Simulation, target *core.Unit)

type WarriorAgent

type WarriorAgent interface {
	GetWarrior() *Warrior
}

Agent is a generic way to access underlying warrior on any of the agents.

type WarriorInputs

type WarriorInputs struct {
	StanceSnapshot bool
}

Directories

Path Synopsis

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL