Bastion aims to serve as a highly performant, simplisitic, and expandable World of Warcraft data visualization framework.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Bastion/scripts/subtlety.lua

2254 lines
88 KiB

-- local Tinkr, Bastion = ...
-- local SubModulue = Bastion.Module:New('sub')
-- local Evaluator = Tinkr.Util.Evaluator
-- local Player = Bastion.UnitManager:Get('player')
-- local None = Bastion.UnitManager:Get('none')
-- local Target = Bastion.UnitManager:Get('target')
-- local RollTheBones = Bastion.SpellBook:GetSpell(315508)
-- local SliceAndDice = Bastion.SpellBook:GetSpell(315496)
-- local BetweenTheEyes = Bastion.SpellBook:GetSpell(315341)
-- local BladeRush = Bastion.SpellBook:GetSpell(271877)
-- local Vanish = Bastion.SpellBook:GetSpell(1856)
-- local Dispatch = Bastion.SpellBook:GetSpell(2098)
-- local Ambush = Bastion.SpellBook:GetSpell(8676)
-- local Stealth = Bastion.SpellBook:GetSpell(1784)
-- local PistolShot = Bastion.SpellBook:GetSpell(185763)
-- local Opportunity = Bastion.SpellBook:GetSpell(195627)
-- local SinisterStrike = Bastion.SpellBook:GetSpell(193315)
-- local GrandMelee = Bastion.SpellBook:GetSpell(193358)
-- local Broadside = Bastion.SpellBook:GetSpell(193356)
-- local TrueBearing = Bastion.SpellBook:GetSpell(193359)
-- local RuthlessPrecision = Bastion.SpellBook:GetSpell(193357)
-- local SkullAndCrossbones = Bastion.SpellBook:GetSpell(199603)
-- local BuriedTreasure = Bastion.SpellBook:GetSpell(199600)
-- local AdrenalineRush = Bastion.SpellBook:GetSpell(13750)
-- local ShadowDance = Bastion.SpellBook:GetSpell(185313)
-- local ShadowDanceAura = Bastion.SpellBook:GetSpell(185422)
-- local Audacity = Bastion.SpellBook:GetSpell(381845)
-- local Flagellation = Bastion.SpellBook:GetSpell(323654)
-- local Dreadblades = Bastion.SpellBook:GetSpell(343142)
-- local JollyRoger = Bastion.SpellBook:GetSpell(199603)
-- local BladeFlurry = Bastion.SpellBook:GetSpell(13877)
-- local Kick = Bastion.SpellBook:GetSpell(1766)
-- local MarkedForDeath = Bastion.SpellBook:GetSpell(137619)
-- local CrimsonVial = Bastion.SpellBook:GetSpell(185311)
-- local Shiv = Bastion.SpellBook:GetSpell(5938)
-- local KidneyShot = Bastion.SpellBook:GetSpell(408)
-- local InstantPoison = Bastion.SpellBook:GetSpell(315584)
-- local Sanguine = Bastion.SpellBook:GetSpell(226512)
-- local AtrophicPosion = Bastion.SpellBook:GetSpell(381637)
-- local Evasion = Bastion.SpellBook:GetSpell(5277)
-- local TricksOfTheTrade = Bastion.SpellBook:GetSpell(57934)
-- local CheapShot = Bastion.SpellBook:GetSpell(1833)
-- local BagOfTricks = Bastion.SpellBook:GetSpell(312411)
-- local AutoAttack = Bastion.SpellBook:GetSpell(6603)
-- local SymbolsOfDeath = Bastion.SpellBook:GetSpell(212283)
-- local ShadowBlades = Bastion.SpellBook:GetSpell(121471)
-- local ColdBlood = Bastion.SpellBook:GetSpell(382245)
-- local ShurikenTornado = Bastion.SpellBook:GetSpell(277925)
-- local ThistleTea = Bastion.SpellBook:GetSpell(381623)
-- local Gloomblade = Bastion.SpellBook:GetSpell(200758)
-- local Shadowstrike = Bastion.SpellBook:GetSpell(185438)
-- local Rupture = Bastion.SpellBook:GetSpell(1943)
-- local Eviscerate = Bastion.SpellBook:GetSpell(196819)
-- local NumbingPoison = Bastion.SpellBook:GetSpell(5761)
-- local ShurikenStorm = Bastion.SpellBook:GetSpell(197835)
-- local BlackPowder = Bastion.SpellBook:GetSpell(319175)
-- local SecretTechnique = Bastion.SpellBook:GetSpell(280719)
-- local DarkBrew = Bastion.SpellBook:GetSpell(310454)
-- local Premeditation = Bastion.SpellBook:GetSpell(343173)
-- local DanseMacabre = Bastion.SpellBook:GetSpell(393969)
-- local IrideusFragment = Bastion.ItemBook:GetItem(193743)
-- local Healthstone = Bastion.ItemBook:GetItem(5512)
-- local WindscarWhetstone = Bastion.ItemBook:GetItem(137486)
-- local DarkMoonRime = Bastion.ItemBook:GetItem(198477)
-- local AlgetharsPuzzleBox = Bastion.ItemBook:GetItem(193701)
-- local RimeCards = {
-- One = Bastion.SpellBook:GetSpell(382844),
-- Two = Bastion.SpellBook:GetSpell(382845),
-- Three = Bastion.SpellBook:GetSpell(382846),
-- Four = Bastion.SpellBook:GetSpell(382847),
-- Five = Bastion.SpellBook:GetSpell(382848),
-- Six = Bastion.SpellBook:GetSpell(382849),
-- Seven = Bastion.SpellBook:GetSpell(382850),
-- Eight = Bastion.SpellBook:GetSpell(382851),
-- }
-- local PurgeTarget = Bastion.UnitManager:CreateCustomUnit('purge', function(unit)
-- local purge = nil
-- Bastion.UnitManager:EnumEnemies(function(unit)
-- if unit:IsDead() then
-- return false
-- end
-- if not Player:CanSee(unit) then
-- return false
-- end
-- if Player:GetDistance(unit) > 40 then
-- return false
-- end
-- if unit:GetAuras():HasAnyStealableAura() then
-- purge = unit
-- return true
-- end
-- end)
-- if purge == nil then
-- purge = None
-- end
-- return purge
-- end)
-- local KickTarget = Bastion.UnitManager:CreateCustomUnit('kick', function(unit)
-- local kick = nil
-- Bastion.UnitManager:EnumEnemies(function(unit)
-- if unit:IsDead() then
-- return false
-- end
-- if not Player:CanSee(unit) then
-- return false
-- end
-- if Player:GetDistance(unit) > 40 then
-- return false
-- end
-- if Player:InMelee(unit) and Player:IsFacing(unit) and Bastion.MythicPlusUtils:CastingCriticalKick(unit, 5) then
-- kick = unit
-- return true
-- end
-- end)
-- if kick == nil then
-- kick = None
-- end
-- return kick
-- end)
-- local Tank = Bastion.UnitManager:CreateCustomUnit('tank', function(unit)
-- local tank = nil
-- Bastion.UnitManager:EnumFriends(function(unit)
-- if Player:GetDistance(unit) > 40 then
-- return false
-- end
-- if not Player:CanSee(unit) then
-- return false
-- end
-- if unit:IsDead() then
-- return false
-- end
-- if unit:IsTank() then
-- tank = unit
-- return true
-- end
-- return false
-- end)
-- if tank == nil then
-- tank = None
-- end
-- return tank
-- end)
-- local RuptureTarget = Bastion.UnitManager:CreateCustomUnit('rupture', function()
-- local target = nil
-- Bastion.UnitManager:EnumEnemies(function(unit)
-- if unit:IsDead() then
-- return false
-- end
-- if not Player:CanSee(unit) then
-- return false
-- end
-- if not Player:InMelee(unit) then
-- return false
-- end
-- if not Player:IsFacing(unit) then
-- return false
-- end
-- if (
-- not unit:GetAuras():FindMy(Rupture):IsUp() or
-- unit:GetAuras():FindMy(Rupture):GetRemainingTime() < 6
-- )
-- and unit:TimeToDie() > 12
-- and unit:GetCombatTime() > 4
-- then
-- target = unit
-- return true
-- end
-- end)
-- if target == nil then
-- target = None
-- end
-- return target
-- end)
-- local DefaultAPL = Bastion.APL:New('default')
-- local AOEAPL = Bastion.APL:New('aoe')
-- local SpecialAPL = Bastion.APL:New('special')
-- local RacialsAPL = Bastion.APL:New('racials')
-- local Facing = function(t)
-- return Bastion.APLTrait:New(function()
-- return Player:IsFacing(t)
-- end)
-- end
-- SpecialAPL:AddSpell(
-- Kick:CastableIf(function(self)
-- return KickTarget:Exists() and self:IsInRange(KickTarget) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling()
-- end):SetTarget(KickTarget)
-- ):AddTraits(
-- Facing(KickTarget)
-- )
-- SpecialAPL:AddSpell(
-- KidneyShot:CastableIf(function(self)
-- return KickTarget:Exists() and self:IsInRange(KickTarget) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Kick:GetTimeSinceLastCast() > 2 and
-- (Player:GetComboPoints(Target) >= 5 or
-- (
-- Player:GetComboPoints(Target) >= 4 and
-- (Player:GetAuras():FindMy(Broadside):IsUp() or Player:GetAuras():FindMy(Opportunity):IsUp())))
-- and not Target:GetAuras():Find(Sanguine):IsUp()
-- end):SetTarget(KickTarget)
-- )
-- SpecialAPL:AddSpell(
-- CheapShot:CastableIf(function(self)
-- return KickTarget:Exists() and self:IsInRange(KickTarget) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and Player:GetAuras():FindMy(Stealth):IsUp()
-- and not Target:GetAuras():Find(Sanguine):IsUp()
-- end):SetTarget(KickTarget)
-- )
-- SpecialAPL:AddSpell(
-- Stealth:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and not Player:IsAffectingCombat() and
-- not Player:GetAuras():FindMy(Stealth):IsUp() and not IsMounted()
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddSpell(
-- CrimsonVial:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetHealthPercent() < 70
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddSpell(
-- Shiv:CastableIf(function(self)
-- return PurgeTarget:Exists() and self:IsInRange(PurgeTarget) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and PurgeTarget:GetAuras():HasAnyStealableAura()
-- end):SetTarget(PurgeTarget)
-- )
-- SpecialAPL:AddSpell(
-- InstantPoison:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- not Player:GetAuras():FindMy(InstantPoison):IsUp() and not Player:IsMoving()
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddSpell(
-- AtrophicPosion:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- not Player:GetAuras():FindMy(AtrophicPosion):IsUp() and not Player:IsMoving()
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddSpell(
-- NumbingPoison:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- not Player:GetAuras():FindMy(NumbingPoison):IsUp() and not Player:IsMoving()
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddItem(
-- Healthstone:UsableIf(function(self)
-- return self:IsEquippedAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetHealthPercent() < 40
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddSpell(
-- TricksOfTheTrade:CastableIf(function(self)
-- return Tank:Exists() and self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:IsTanking(Target)
-- end):SetTarget(Tank)
-- )
-- SpecialAPL:AddSpell(
-- Evasion:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetHealthPercent() < 40
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddItem(
-- IrideusFragment:UsableIf(function(self)
-- return self:IsEquippedAndUsable() and
-- not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 2 or Target:IsBoss())
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddItem(
-- WindscarWhetstone:UsableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and self:IsEquippedAndUsable() and
-- not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 2 or Target:IsBoss())
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddItem(
-- AlgetharsPuzzleBox:UsableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and self:IsEquippedAndUsable() and
-- not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 3 or Target:IsBoss())
-- end):SetTarget(Player)
-- )
-- SpecialAPL:AddItem(
-- DarkMoonRime:UsableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and self:IsEquippedAndUsable() and
-- not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 2 or Target:IsBoss()) and
-- (Player:GetAuras():FindMy(RimeCards.One):IsUp() or
-- Player:GetAuras():FindMy(RimeCards.Two):IsUp() or
-- Player:GetAuras():FindMy(RimeCards.Three):IsUp() or
-- Player:GetAuras():FindMy(RimeCards.Four):IsUp() or
-- Player:GetAuras():FindMy(RimeCards.Five):IsUp() or
-- Player:GetAuras():FindMy(RimeCards.Six):IsUp() or
-- Player:GetAuras():FindMy(RimeCards.Seven):IsUp() or
-- Player:GetAuras():FindMy(RimeCards.Eight):IsUp()
-- )
-- end):SetTarget(Target)
-- )
-- -- Use Shadowstrike during Shadow Dance.
-- SpecialAPL:AddSpell(
-- Shadowstrike:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and Player:GetAuras():FindMy(Premeditation):IsUp() and
-- Player:GetEnemies(10) <= 3
-- end):SetTarget(Target)
-- )
-- RacialsAPL:AddSpell(
-- BagOfTricks:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling()
-- end):SetTarget(Target)
-- )
-- -- Use Symbols of Death on cooldown as much as possible.
-- DefaultAPL:AddSpell(
-- SymbolsOfDeath:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling()
-- end):SetTarget(Player):OnCast(function()
-- ShurikenTornado:Cast(Target)
-- end)
-- )
-- -- Use Shadow Blades on cooldown.
-- DefaultAPL:AddSpell(
-- ShadowBlades:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling()
-- end):SetTarget(Player)
-- )
-- -- Use Cold Blood before a finishing move, ideally before Secret Technique.
-- DefaultAPL:AddSpell(
-- ColdBlood:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetComboPoints(Target) >= 5 and SecretTechnique:IsKnownAndUsable() and
-- Player:GetAuras():FindMy(SliceAndDice):IsUp() and
-- Target:GetAuras():FindMy(Rupture):IsUp()
-- end):SetTarget(Player):OnCast(function()
-- SecretTechnique:Cast(Target)
-- end)
-- )
-- -- Line up Shuriken Tornado with Symbols of Death.
-- DefaultAPL:AddSpell(
-- ShurikenTornado:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetAuras():FindMy(SymbolsOfDeath):IsUp()
-- end):SetTarget(Player)
-- )
-- -- Use Shadow Dance on cooldown as much as possible.
-- DefaultAPL:AddSpell(
-- ShadowDance:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and Gloomblade:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and Player:GetComboPoints(Target) <= 2
-- end):SetTarget(Player):OnCast(function()
-- Gloomblade:Cast(Target) -- We want to cast gloomblade immediately with shadow dance to trigger 1 stack of danse macabre
-- end)
-- )
-- -- Use Thistle Tea when low on energy.
-- -- actions.cds+=/thistle_tea,if=cooldown.symbols_of_death.remains>=3&!buff.thistle_tea.up&(energy.deficit>=100|cooldown.thistle_tea.charges_fractional>=2.75&buff.shadow_dance.up)|buff.shadow_dance.remains>=4&!buff.thistle_tea.up&spell_targets.shuriken_storm>=3|!buff.thistle_tea.up&fight_remains<=(6*cooldown.thistle_tea.charges)
-- DefaultAPL:AddSpell(
-- ThistleTea:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetPowerDeficit() >= 100 and
-- ThistleTea:GetTimeSinceLastCast() >= 3
-- end):SetTarget(Player)
-- )
-- -- Use Finishing moves with 6 or more combo points (5 or more during Shadow Dance) with the following priority:
-- -- Cast Slice and Dice if it needs to be refreshed for maintenance or if it is not up.
-- DefaultAPL:AddSpell(
-- SliceAndDice:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 6 or
-- (Player:GetComboPoints(Target) >= 5 and
-- Player:GetAuras():FindMy(ShadowDanceAura):IsUp())) and
-- (
-- not Player:GetAuras():FindMy(SliceAndDice):IsUp() or
-- Player:GetAuras():FindMy(SliceAndDice):GetRemainingTime() < 12
-- )
-- end):SetTarget(Player)
-- )
-- -- Cast Rupture if it needs to be refreshed for maintenance or if it is not up.
-- DefaultAPL:AddSpell(
-- Rupture:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 6 or
-- (Player:GetComboPoints(Target) >= 5 and
-- Player:GetAuras():FindMy(ShadowDanceAura):IsUp())) and (
-- not Target:GetAuras():FindMy(Rupture):IsUp() or
-- Target:GetAuras():FindMy(Rupture):GetRemainingTime() < 6
-- ) and not Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
-- end):SetTarget(Target)
-- )
-- DefaultAPL:AddSpell(
-- SecretTechnique:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 5) and
-- Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
-- (Player:GetAuras():FindMy(DanseMacabre):GetCount() >= 3 or
-- not DanseMacabre:IsKnown()) and
-- (not ColdBlood:IsKnown() or
-- ColdBlood:GetCooldownRemaining() > Player:GetAuras():FindMy(ShadowDanceAura):GetRemainingTime() - 2)
-- end):SetTarget(Target)
-- )
-- -- Cast Eviscerate if it is available.
-- DefaultAPL:AddSpell(
-- Eviscerate:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 6 or
-- (Player:GetComboPoints(Target) >= 5 and
-- Player:GetAuras():FindMy(ShadowDanceAura):IsUp()))
-- end):SetTarget(Target)
-- )
-- -- Vanish - Is a fairly weak cooldown. It is best to use on low combo points for a Shadowstrike cast. Use it after Secret Technique in Shadow Dance when playing with Danse Macabre.
-- DefaultAPL:AddSpell(
-- Vanish:CastableIf(function(self)
-- return Tank:Exists() and Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetComboPoints(Target) < 4
-- end):SetTarget(Player)
-- )
-- -- Use Combo Point builder with the following priority:
-- -- Use Gloomblade outside of Shadow Dance.
-- DefaultAPL:AddSpell(
-- Gloomblade:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- not Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
-- end):SetTarget(Target)
-- )
-- -- Use Shadowstrike during Shadow Dance.
-- DefaultAPL:AddSpell(
-- Shadowstrike:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
-- end):SetTarget(Target)
-- )
-- -- AOE
-- -- Use Symbols of Death on cooldown as much as possible.
-- AOEAPL:AddSpell(
-- SymbolsOfDeath:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling()
-- end):SetTarget(Player):OnCast(function()
-- ShurikenTornado:Cast(Target)
-- end)
-- )
-- -- Use Shadow Blades on cooldown.
-- AOEAPL:AddSpell(
-- ShadowBlades:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling()
-- end):SetTarget(Player)
-- )
-- -- Use Cold Blood before a finishing move.
-- AOEAPL:AddSpell(
-- ColdBlood:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetComboPoints(Target) >= 5 and SecretTechnique:IsKnownAndUsable() and
-- Player:GetAuras():FindMy(SliceAndDice):IsUp() and
-- Target:GetAuras():FindMy(Rupture):IsUp()
-- end):SetTarget(Player):OnCast(function()
-- SecretTechnique:Cast(Target)
-- end)
-- )
-- -- Line up Shuriken Tornado with Symbols of Death.
-- AOEAPL:AddSpell(
-- ShurikenTornado:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetAuras():FindMy(SymbolsOfDeath):IsUp()
-- end):SetTarget(Target)
-- )
-- -- Use Shadow Dance on cooldown as much as possible.
-- AOEAPL:AddSpell(
-- ShadowDance:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and Gloomblade:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and Player:GetComboPoints(Target) <= 2
-- end):SetTarget(Player):OnCast(function()
-- Gloomblade:Cast(Target) -- We want to cast gloomblade immediately with shadow dance to trigger 1 stack of danse macabre
-- end)
-- )
-- -- Use Thistle Tea with Shadow Dance.
-- AOEAPL:AddSpell(
-- ThistleTea:CastableIf(function(self)
-- return Target:Exists() and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetPowerDeficit() >= 100 and
-- ThistleTea:GetTimeSinceLastCast() >= 3
-- end):SetTarget(Player)
-- )
-- -- Use Finishing moves with 5 or more combo points with the following priority:
-- -- Cast Slice and Dice if it needs to be refreshed for maintenance or if it is not up.
-- AOEAPL:AddSpell(
-- SliceAndDice:CastableIf(function(self)
-- return Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 5) and
-- (
-- not Player:GetAuras():FindMy(SliceAndDice):IsUp() or
-- Player:GetAuras():FindMy(SliceAndDice):GetRemainingTime() < 6
-- )
-- and Player:GetEnemies(10) < 6
-- end):SetTarget(Target)
-- )
-- -- Cast Rupture if it needs to be refreshed for maintenance or if it is not up.
-- AOEAPL:AddSpell(
-- Rupture:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 5) and (
-- not Target:GetAuras():FindMy(Rupture):IsUp() or
-- Target:GetAuras():FindMy(Rupture):GetRemainingTime() < 6
-- )
-- and not Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
-- and not Player:GetAuras():FindMy(SymbolsOfDeath):IsUp()
-- and not Player:GetAuras():FindMy(ThistleTea):IsUp()
-- end):SetTarget(Target)
-- )
-- -- Cast Rupture on all targets. (scam??)
-- AOEAPL:AddSpell(
-- Rupture:CastableIf(function(self)
-- return RuptureTarget:Exists() and self:IsInRange(RuptureTarget) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(RuptureTarget) >= 6) and (
-- not RuptureTarget:GetAuras():FindMy(Rupture):IsUp() or
-- RuptureTarget:GetAuras():FindMy(Rupture):GetRemainingTime() < 6
-- )
-- and not Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
-- end):SetTarget(RuptureTarget)
-- )
-- -- actions.finish+=/secret_technique,if=buff.shadow_dance.up&(buff.danse_macabre.stack>=3|!talent.danse_macabre)&(!talent.cold_blood|cooldown.cold_blood.remains>buff.shadow_dance.remains-2)
-- AOEAPL:AddSpell(
-- SecretTechnique:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 5) and
-- Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
-- (Player:GetAuras():FindMy(DanseMacabre):GetCount() >= 3 or
-- not DanseMacabre:IsKnown()) and
-- (not ColdBlood:IsKnown() or
-- ColdBlood:GetCooldownRemaining() > Player:GetAuras():FindMy(ShadowDanceAura):GetRemainingTime() - 2)
-- end):SetTarget(Target)
-- )
-- -- Cast Black Powder with 3 or more targets, 2 or more when talented into Dark Brew.
-- AOEAPL:AddSpell(
-- BlackPowder:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- (Player:GetComboPoints(Target) >= 5) and
-- (Player:GetEnemies(10) >= 3 or
-- (Player:GetEnemies(10) >= 2 and
-- DarkBrew:IsKnown()))
-- end):SetTarget(Target)
-- )
-- -- Cast Eviscerate.
-- AOEAPL:AddSpell(
-- Eviscerate:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetComboPoints(Target) >= 5
-- end):SetTarget(Target)
-- )
-- -- Vanish - Is a fairly weak cooldown. It is best to use on low combo points for a Shadowstrike cast. Use it after Secret Technique in Shadow Dance when playing with Danse Macabre.
-- AOEAPL:AddSpell(
-- Vanish:CastableIf(function(self)
-- return Tank:Exists() and Target:Exists() and Player:InMelee(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetComboPoints(Target) < 4
-- end):SetTarget(Player)
-- )
-- -- Use Combo Point builder with the following priority:
-- -- Use Shuriken Storm on 2 targets outside of Shadow Dance.
-- AOEAPL:AddSpell(
-- ShurikenStorm:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetEnemies(10) == 2 and
-- not Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
-- end):SetTarget(Player)
-- )
-- -- Use Shadowstrike on 2 and 3 targets during Shadow Dance or to proc Premeditation.
-- AOEAPL:AddSpell(
-- Shadowstrike:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetEnemies(10) >= 2 and Player:GetEnemies(10) <= 3 and
-- Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
-- end):SetTarget(Target)
-- )
-- -- Use Shuriken Storm at > 2 targets.
-- AOEAPL:AddSpell(
-- ShurikenStorm:CastableIf(function(self)
-- return self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetEnemies(10) > 2
-- end):SetTarget(Player)
-- )
-- -- Use gloomblade at <= 2 targets.
-- AOEAPL:AddSpell(
-- Gloomblade:CastableIf(function(self)
-- return Target:Exists() and self:IsInRange(Target) and
-- self:IsKnownAndUsable() and
-- not Player:IsCastingOrChanneling() and
-- Player:GetEnemies(10) <= 2
-- end):SetTarget(Target)
-- )
-- SubModulue:Sync(function()
-- SpecialAPL:Execute()
-- if Player:GetEnemies(10) >= 2 then
-- AOEAPL:Execute()
-- else
-- DefaultAPL:Execute()
-- end
-- RacialsAPL:Execute()
-- end)
-- Bastion:Register(SubModulue)
local Tinkr, Bastion = ...
local SubModulue = Bastion.Module:New('sub')
local Evaluator = Tinkr.Util.Evaluator
local Player = Bastion.UnitManager:Get('player')
local None = Bastion.UnitManager:Get('none')
local Target = Bastion.UnitManager:Get('target')
Player:WatchForSwings()
local RollTheBones = Bastion.SpellBook:GetSpell(315508)
local SliceAndDice = Bastion.SpellBook:GetSpell(315496)
local BetweenTheEyes = Bastion.SpellBook:GetSpell(315341)
local BladeRush = Bastion.SpellBook:GetSpell(271877)
local Vanish = Bastion.SpellBook:GetSpell(1856)
local Dispatch = Bastion.SpellBook:GetSpell(2098)
local Ambush = Bastion.SpellBook:GetSpell(8676)
local FlagellationPersist = Bastion.SpellBook:GetSpell(345569)
local Stealth = Bastion.SpellBook:GetSpell(1784)
local PistolShot = Bastion.SpellBook:GetSpell(185763)
local Opportunity = Bastion.SpellBook:GetSpell(195627)
local SinisterStrike = Bastion.SpellBook:GetSpell(193315)
local GrandMelee = Bastion.SpellBook:GetSpell(193358)
local Broadside = Bastion.SpellBook:GetSpell(193356)
local TrueBearing = Bastion.SpellBook:GetSpell(193359)
local RuthlessPrecision = Bastion.SpellBook:GetSpell(193357)
local DeviousStratagem = Bastion.SpellBook:GetSpell(193531)
local SkullAndCrossbones = Bastion.SpellBook:GetSpell(199603)
local ShadowFocus = Bastion.SpellBook:GetSpell(108209)
local BuriedTreasure = Bastion.SpellBook:GetSpell(199600)
local AdrenalineRush = Bastion.SpellBook:GetSpell(13750)
local ShadowDance = Bastion.SpellBook:GetSpell(185313)
local ShadowDanceAura = Bastion.SpellBook:GetSpell(185422)
local Shadowmeld = Bastion.SpellBook:GetSpell(58984)
local Audacity = Bastion.SpellBook:GetSpell(381845)
local SealFate = Bastion.SpellBook:GetSpell(14190)
local Flagellation = Bastion.SpellBook:GetSpell(323654)
local Dreadblades = Bastion.SpellBook:GetSpell(343142)
local MasterOfShadows = Bastion.SpellBook:GetSpell(196976)
local JollyRoger = Bastion.SpellBook:GetSpell(199603)
local BladeFlurry = Bastion.SpellBook:GetSpell(13877)
local Kick = Bastion.SpellBook:GetSpell(1766)
local FinalityRupture = Bastion.SpellBook:GetSpell(385951)
local MarkedForDeath = Bastion.SpellBook:GetSpell(137619)
local CrimsonVial = Bastion.SpellBook:GetSpell(185311)
local TheRotten = Bastion.SpellBook:GetSpell(394203)
local Shiv = Bastion.SpellBook:GetSpell(5938)
local KidneyShot = Bastion.SpellBook:GetSpell(408)
local InstantPoison = Bastion.SpellBook:GetSpell(315584)
local Sanguine = Bastion.SpellBook:GetSpell(226512)
local AtrophicPosion = Bastion.SpellBook:GetSpell(381637)
local Evasion = Bastion.SpellBook:GetSpell(5277)
local TricksOfTheTrade = Bastion.SpellBook:GetSpell(57934)
local Backstab = Bastion.SpellBook:GetSpell(53)
local CheapShot = Bastion.SpellBook:GetSpell(1833)
local BagOfTricks = Bastion.SpellBook:GetSpell(312411)
local Alacrity = Bastion.SpellBook:GetSpell(193539)
local PerforatedVeins = Bastion.SpellBook:GetSpell(394254)
local AutoAttack = Bastion.SpellBook:GetSpell(6603)
local DeeperStratagem = Bastion.SpellBook:GetSpell(193531)
local SecretStratagem = Bastion.SpellBook:GetSpell(394320)
local SymbolsOfDeath = Bastion.SpellBook:GetSpell(212283)
local ShadowBlades = Bastion.SpellBook:GetSpell(121471)
local Vigor = Bastion.SpellBook:GetSpell(14983)
local ColdBlood = Bastion.SpellBook:GetSpell(382245)
local ShurikenTornado = Bastion.SpellBook:GetSpell(277925)
local ThistleTea = Bastion.SpellBook:GetSpell(381623)
local Gloomblade = Bastion.SpellBook:GetSpell(200758)
local Shadowstrike = Bastion.SpellBook:GetSpell(185438)
local Rupture = Bastion.SpellBook:GetSpell(1943)
local Eviscerate = Bastion.SpellBook:GetSpell(196819)
local ResoundingClarity = Bastion.SpellBook:GetSpell(381622)
local ArcanePulse = Bastion.SpellBook:GetSpell(260364)
local NumbingPoison = Bastion.SpellBook:GetSpell(5761)
local ShurikenStorm = Bastion.SpellBook:GetSpell(197835)
local BlackPowder = Bastion.SpellBook:GetSpell(319175)
local Sepsis = Bastion.SpellBook:GetSpell(385408)
local SecretTechnique = Bastion.SpellBook:GetSpell(280719)
local DarkBrew = Bastion.SpellBook:GetSpell(310454)
local Premeditation = Bastion.SpellBook:GetSpell(343173)
local ArcaneTorrent = Bastion.SpellBook:GetSpell(50613)
local DanseMacabre = Bastion.SpellBook:GetSpell(393969)
local LingeringShadow = Bastion.SpellBook:GetSpell(385960)
local EchoingReprimand = Bastion.SpellBook:GetSpell(385616)
local LightsJudgment = Bastion.SpellBook:GetSpell(255647)
local Subterfuge = Bastion.SpellBook:GetSpell(108208)
local EchoingReprimand2 = Bastion.SpellBook:GetSpell(323558)
local EchoingReprimand3 = Bastion.SpellBook:GetSpell(323559)
local EchoingReprimand4 = Bastion.SpellBook:GetSpell(323560)
local EchoingReprimand5 = Bastion.SpellBook:GetSpell(354838)
local SilentStorm = Bastion.SpellBook:GetSpell(385722)
local FindWeakness = Bastion.SpellBook:GetSpell(91023)
local ImprovedShurikenStorm = Bastion.SpellBook:GetSpell(319951)
local RefreshingHealingPotion = Bastion.ItemBook:GetItem(191380)
local ElementalPotionOfPower = Bastion.ItemBook:GetItem(191389)
local IrideusFragment = Bastion.ItemBook:GetItem(193743)
local Healthstone = Bastion.ItemBook:GetItem(5512)
local WindscarWhetstone = Bastion.ItemBook:GetItem(137486)
local DarkMoonRime = Bastion.ItemBook:GetItem(198477)
local AlgetharsPuzzleBox = Bastion.ItemBook:GetItem(193701)
local RimeCards = {
Two = Bastion.SpellBook:GetSpell(382845),
Three = Bastion.SpellBook:GetSpell(382846),
Four = Bastion.SpellBook:GetSpell(382847),
Five = Bastion.SpellBook:GetSpell(382848),
Six = Bastion.SpellBook:GetSpell(382849),
Seven = Bastion.SpellBook:GetSpell(382850),
Eight = Bastion.SpellBook:GetSpell(382851),
Ace = Bastion.SpellBook:GetSpell(382844),
}
local PurgeTarget = Bastion.UnitManager:CreateCustomUnit('purge', function(unit)
local purge = nil
Bastion.UnitManager:EnumEnemies(function(unit)
if unit:IsDead() then
return false
end
if not Player:CanSee(unit) then
return false
end
if Player:GetDistance(unit) > 40 then
return false
end
if unit:GetAuras():HasAnyStealableAura() and Shiv:IsInRange(unit) then
purge = unit
return true
end
end)
if purge == nil then
purge = None
end
return purge
end)
local KickTarget = Bastion.UnitManager:CreateCustomUnit('kick', function(unit)
local kick = nil
Bastion.UnitManager:EnumEnemies(function(unit)
if unit:IsDead() then
return false
end
if not Player:CanSee(unit) then
return false
end
if Player:GetDistance(unit) > 40 then
return false
end
if Player:InMelee(unit) and Player:IsFacing(unit) and Bastion.MythicPlusUtils:CastingCriticalKick(unit, 5) then
kick = unit
return true
end
end)
if kick == nil then
kick = None
end
return kick
end)
local Tank = Bastion.UnitManager:CreateCustomUnit('tank', function(unit)
local tank = nil
Bastion.UnitManager:EnumFriends(function(unit)
if Player:GetDistance(unit) > 40 then
return false
end
if not Player:CanSee(unit) then
return false
end
if unit:IsDead() then
return false
end
if unit:IsTank() then
tank = unit
return true
end
return false
end)
if tank == nil then
tank = None
end
return tank
end)
local RuptureTarget = Bastion.UnitManager:CreateCustomUnit('rupture', function()
local target = nil
Bastion.UnitManager:EnumEnemies(function(unit)
if unit:IsDead() then
return false
end
if not Player:CanSee(unit) then
return false
end
if not Player:InMelee(unit) then
return false
end
if not Player:IsFacing(unit) then
return false
end
if (
not unit:GetAuras():FindMy(Rupture):IsUp() or
unit:GetAuras():FindMy(Rupture):GetRemainingTime() < 6
)
and unit:TimeToDie() > 12
and unit:GetCombatTime() > 4
then
target = unit
return true
end
end)
if target == nil then
target = None
end
return target
end)
local function GetRimeDuration(topCard)
-- // card order is [ 2 3 4 5 6 7 8 A ]
if topCard == RimeCards.Two then
return 0
elseif topCard == RimeCards.Three then
return 1
elseif topCard == RimeCards.Four then
return 2
elseif topCard == RimeCards.Five then
return 3
elseif topCard == RimeCards.Six then
return 4
elseif topCard == RimeCards.Seven then
return 5
elseif topCard == RimeCards.Eight then
return 6
elseif topCard == RimeCards.Ace then
return 7
end
return 0
end
local function GetRimeTopCard()
local topCard = nil
if Player:GetAuras():FindMy(RimeCards.Ace):IsUp() then
topCard = RimeCards.Ace
elseif Player:GetAuras():FindMy(RimeCards.Two):IsUp() then
topCard = RimeCards.Two
elseif Player:GetAuras():FindMy(RimeCards.Three):IsUp() then
topCard = RimeCards.Three
elseif Player:GetAuras():FindMy(RimeCards.Four):IsUp() then
topCard = RimeCards.Four
elseif Player:GetAuras():FindMy(RimeCards.Five):IsUp() then
topCard = RimeCards.Five
elseif Player:GetAuras():FindMy(RimeCards.Six):IsUp() then
topCard = RimeCards.Six
elseif Player:GetAuras():FindMy(RimeCards.Seven):IsUp() then
topCard = RimeCards.Seven
elseif Player:GetAuras():FindMy(RimeCards.Eight):IsUp() then
topCard = RimeCards.Eight
end
return topCard
end
local function HasRimeCard()
return GetRimeTopCard() ~= nil
end
local RimeTarget = Bastion.UnitManager:CreateCustomUnit('rime', function()
local target = nil
Bastion.UnitManager:EnumEnemies(function(unit)
if unit:IsDead() then
return false
end
if not Player:CanSee(unit) then
return false
end
if not Player:InMelee(unit) then
return false
end
if not Player:IsFacing(unit) then
return false
end
if not unit:IsBoss() and Player:GetEnemies(10) < 3 then
return false
end
if HasRimeCard() and DarkMoonRime:IsEquippedAndUsable() and
(unit:TimeToDie() < GetRimeDuration(GetRimeTopCard()) - 0.5 or unit:IsBoss()) then
target = unit
return true
end
end)
if target == nil then
target = None
end
return target
end)
local DefaultAPL = Bastion.APL:New('default')
local CDsAPL = Bastion.APL:New('cds')
local StealthedAPL = Bastion.APL:New('stealthed')
local StealthCDsAPL = Bastion.APL:New('stealthed_cds')
local FinishAPL = Bastion.APL:New('finish')
local BuildAPL = Bastion.APL:New('build')
local ItemsAPL = Bastion.APL:New('items')
ItemsAPL:AddItem(
Healthstone:UsableIf(function(self)
return self:IsEquippedAndUsable() and
not Player:IsCastingOrChanneling() and
Player:GetHealthPercent() < 40
end):SetTarget(Player)
)
ItemsAPL:AddItem(
RefreshingHealingPotion:UsableIf(function(self)
return self:IsEquippedAndUsable() and
not Player:IsCastingOrChanneling() and
Player:GetHealthPercent() < 40
end):SetTarget(Player)
)
ItemsAPL:AddSpell(
TricksOfTheTrade:CastableIf(function(self)
return Tank:Exists() and Target:Exists() and self:IsKnownAndUsable() and
not Player:IsCastingOrChanneling() and
Player:IsTanking(Target)
end):SetTarget(Tank)
)
ItemsAPL:AddSpell(
Evasion:CastableIf(function(self)
return self:IsKnownAndUsable() and
not Player:IsCastingOrChanneling() and
Player:GetHealthPercent() < 40
end):SetTarget(Player)
)
ItemsAPL:AddItem(
IrideusFragment:UsableIf(function(self)
return self:IsEquippedAndUsable() and
not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 2 or Target:IsBoss())
end):SetTarget(Player)
)
ItemsAPL:AddItem(
ElementalPotionOfPower:UsableIf(function(self)
return self:IsEquippedAndUsable() and
not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 2 or Target:IsBoss())
end):SetTarget(Player)
)
ItemsAPL:AddItem(
WindscarWhetstone:UsableIf(function(self)
return Target:Exists() and Player:InMelee(Target) and self:IsEquippedAndUsable() and
not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 2 or Target:IsBoss())
end):SetTarget(Player)
)
ItemsAPL:AddItem(
AlgetharsPuzzleBox:UsableIf(function(self)
return Target:Exists() and Player:InMelee(Target) and self:IsEquippedAndUsable() and
not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 3 or Target:IsBoss()) and
not Player:IsMoving()
end):SetTarget(Player)
)
-- Since we are sniping soon to die targets with rime we can put it in the default APL since we want burst damage
DefaultAPL:AddItem(
DarkMoonRime:UsableIf(function(self)
return RimeTarget:Exists() and Player:InMelee(RimeTarget) and self:IsEquippedAndUsable() and
not Player:IsCastingOrChanneling() and (Player:GetMeleeAttackers() > 2 or Target:IsBoss()) and
HasRimeCard()
end):SetTarget(RimeTarget)
)
-- # Executed every time the actor is available.
-- # Restealth if possible (no vulnerable enemies in combat)
-- actions=stealth
DefaultAPL:AddSpell(
Stealth:CastableIf(
function(self)
return self:IsKnownAndUsable() and not Player:GetAuras():FindMy(Stealth):IsUp() and
not Player:IsAffectingCombat() and not IsMounted()
end
):SetTarget(Player)
)
-- # Interrupt on cooldown to allow simming interactions with that
-- actions+=/kick
DefaultAPL:AddSpell(
Kick:CastableIf(function(self)
return KickTarget:Exists() and self:IsInRange(KickTarget) and
self:IsKnownAndUsable() and
not Player:IsCastingOrChanneling() and Player:IsFacing(Target)
end):SetTarget(KickTarget)
)
DefaultAPL:AddSpell(
KidneyShot:CastableIf(function(self)
return KickTarget:Exists() and self:IsInRange(KickTarget) and
self:IsKnownAndUsable() and
not Player:IsCastingOrChanneling() and Player:IsFacing(Target)
end):SetTarget(KickTarget)
)
-- Purge
DefaultAPL:AddSpell(
Shiv:CastableIf(function(self)
return PurgeTarget:Exists() and
self:IsKnownAndUsable() and
not Player:IsCastingOrChanneling() and Player:IsFacing(Target)
end):SetTarget(PurgeTarget)
)
-- double consume_cp_max() const
-- {
-- return COMBO_POINT_MAX + as<double>( talent.rogue.deeper_stratagem->effectN( 2 ).base_value() +
-- talent.outlaw.devious_stratagem->effectN( 2 ).base_value() +
-- talent.subtlety.secret_stratagem->effectN( 2 ).base_value() );
-- }
-- # Used to determine whether cooldowns wait for SnD based on targets.
-- actions+=/variable,name=snd_condition,value=buff.slice_and_dice.up|spell_targets.shuriken_storm>=cp_max_spend
DefaultAPL:AddVariable(
'snd_condition',
function()
return Player:GetAuras():FindMy(SliceAndDice):IsUp() or
Player:GetEnemies(10) >= ConsumeCPMax()
end
)
-- # Check to see if the next CP (in the event of a ShT proc) is Animacharged
-- actions+=/variable,name=is_next_cp_animacharged,if=talent.echoing_reprimand.enabled,value=combo_points=1&buff.echoing_reprimand_2.up|combo_points=2&buff.echoing_reprimand_3.up|combo_points=3&buff.echoing_reprimand_4.up|combo_points=4&buff.echoing_reprimand_5.up
DefaultAPL:AddVariable(
'is_next_cp_animacharged',
function()
if not EchoingReprimand:IsKnown() then
return false
end
local comboPoints = Player:GetComboPoints()
if comboPoints == 1 and Player:GetAuras():FindMy(EchoingReprimand2):IsUp() then
return true
end
if comboPoints == 2 and Player:GetAuras():FindMy(EchoingReprimand3):IsUp() then
return true
end
if comboPoints == 3 and Player:GetAuras():FindMy(EchoingReprimand4):IsUp() then
return true
end
if comboPoints == 4 and Player:GetAuras():FindMy(EchoingReprimand5):IsUp() then
return true
end
return false
end
)
-- # Account for ShT reaction time by ignoring low-CP animacharged matches in the 0.5s preceeding a potential ShT proc
-- actions+=/variable,name=effective_combo_points,value=effective_combo_points
DefaultAPL:AddVariable(
'effective_combo_points',
function()
local cur = Player:GetComboPoints() or 0
if not EchoingReprimand:IsKnown() then
return cur
end
if cur < 2 or cur > 5 then
return cur
end
if Player:GetAuras():FindMy(EchoingReprimand):IsUp() or Player:GetAuras():FindMy(EchoingReprimand2):IsUp() or
Player:GetAuras():FindMy(EchoingReprimand3):IsUp() or
Player:GetAuras():FindMy(EchoingReprimand4):IsUp() or
Player:GetAuras():FindMy(EchoingReprimand5):IsUp()
then
return 7
end
return cur
end
)
-- actions+=/variable,name=effective_combo_points,if=talent.echoing_reprimand.enabled&effective_combo_points>combo_points&combo_points.deficit>2&time_to_sht.4.plus<0.5&!variable.is_next_cp_animacharged,value=combo_points
DefaultAPL:AddVariable(
'effective_combo_points',
function()
if not EchoingReprimand:IsKnown() then
return 0
end
local cur = Player:GetComboPoints() or 0
local deficit = Player:GetComboPointsDeficit() or 0
if cur > Player:GetComboPoints() and deficit > 2 and
Player:GetAuras():FindMy(EchoingReprimand4):GetRemainingTime() < 0.5 and
not DefaultAPL:GetVariable('is_next_cp_animacharged')
then
return cur
end
return 0
end
)
-- # Check CDs at first
-- actions+=/call_action_list,name=cds
DefaultAPL:AddAPL(
CDsAPL,
function()
return true
end
)
-- # Apply Slice and Dice at 4+ CP if it expires within the next GCD or is not up
-- actions+=/slice_and_dice,if=spell_targets.shuriken_storm<cp_max_spend&buff.slice_and_dice.remains<gcd.max&fight_remains>6&combo_points>=4
DefaultAPL:AddSpell(
SliceAndDice:CastableIf(
function(self)
return self:IsKnownAndUsable() and Player:GetEnemies(10) < ConsumeCPMax() and
Player:GetAuras():FindMy(SliceAndDice):GetRemainingTime() < Player:GetGCD() and
Player:GetAuras():FindMy(SliceAndDice):GetRemainingTime() > 6 and
Player:GetComboPoints() >= 4
end
):SetTarget(Player)
)
-- # Run fully switches to the Stealthed Rotation (by doing so, it forces pooling if nothing is available).
-- actions+=/run_action_list,name=stealthed,if=stealthed.all
DefaultAPL:AddAPL(
StealthedAPL,
function()
return Player:GetAuras():FindMy(Stealth):IsUp() or Player:GetAuras():FindMy(Subterfuge):IsUp() or
Player:GetAuras():FindMy(ShadowDanceAura):IsUp()
end
)
-- # Only change rotation if we have priority_rotation set.
-- actions+=/variable,name=priority_rotation,value=priority_rotation
DefaultAPL:AddVariable(
'priority_rotation',
function()
return false
end
)
-- # Used to define when to use stealth CDs or builders
-- actions+=/variable,name=stealth_threshold,value=25+talent.vigor.enabled*20+talent.master_of_shadows.enabled*20+talent.shadow_focus.enabled*25+talent.alacrity.enabled*20+25*(spell_targets.shuriken_storm>=4)
DefaultAPL:AddVariable(
'stealth_threshold',
function()
return 25 + (Vigor:IsKnown() and 20 or 0) + (MasterOfShadows:IsKnown() and 20 or 0) +
(ShadowFocus:IsKnown() and 25 or 0) + (Alacrity:IsKnown() and 20 or 0) +
(25 * (Player:GetEnemies(10) >= 4 and 1 or 0))
end
)
-- # Consider using a Stealth CD when reaching the energy threshold
-- actions+=/call_action_list,name=stealth_cds,if=energy.deficit<=variable.stealth_threshold
DefaultAPL:AddAPL(
StealthCDsAPL,
function()
return Player:GetPowerDeficit() <= DefaultAPL:GetVariable('stealth_threshold')
end
)
-- actions+=/call_action_list,name=finish,if=variable.effective_combo_points>=cp_max_spend
DefaultAPL:AddAPL(
FinishAPL,
function()
return DefaultAPL:GetVariable('effective_combo_points') >= ConsumeCPMax()
end
)
-- # Finish at maximum or close to maximum combo point value
-- actions+=/call_action_list,name=finish,if=combo_points.deficit<=1+buff.the_rotten.up|fight_remains<=1&variable.effective_combo_points>=3
DefaultAPL:AddAPL(
FinishAPL,
function()
return Player:GetComboPointsDeficit() <= 1 + (((Player:GetAuras():FindMy(TheRotten):IsUp()) or
(Player:TimeToDie() <= 1 and DefaultAPL:GetVariable('effective_combo_points') >= 3)) and 1 or 0)
end
)
-- # Finish at 4+ against 4 targets (outside stealth)
-- actions+=/call_action_list,name=finish,if=spell_targets.shuriken_storm>=(4-talent.seal_fate)&variable.effective_combo_points>=4
DefaultAPL:AddAPL(
FinishAPL,
function()
return Player:GetEnemies(10) >= (4 - (SealFate:IsKnown() and 1 or 0)) and
DefaultAPL:GetVariable('effective_combo_points') >= 4
end
)
-- # Use a builder when reaching the energy threshold
-- actions+=/call_action_list,name=build,if=energy.deficit<=variable.stealth_threshold
DefaultAPL:AddAPL(
BuildAPL,
function()
return Player:GetPowerDeficit() <= DefaultAPL:GetVariable('stealth_threshold')
end
)
-- # Lowest priority in all of the APL because it causes a GCD
-- actions+=/arcane_torrent,if=energy.deficit>=15+energy.regen
DefaultAPL:AddSpell(
ArcaneTorrent:CastableIf(
function(self)
return self:IsKnownAndUsable() and Player:InMelee(Target) and
Player:GetPowerDeficit() >= 15 + Player:GetPowerRegen()
end
):SetTarget(Player)
)
-- actions+=/arcane_pulse
DefaultAPL:AddSpell(
ArcanePulse:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and true
end
):SetTarget(Target)
)
-- actions+=/lights_judgment
DefaultAPL:AddSpell(
LightsJudgment:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and true
end
):SetTarget(Target)
)
-- actions+=/bag_of_tricks
DefaultAPL:AddSpell(
BagOfTricks:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and true
end
):SetTarget(Target)
)
-- # Builders
-- actions.build=shuriken_storm,if=spell_targets>=2+(talent.gloomblade&buff.lingering_shadow.remains>=6|buff.perforated_veins.up)
BuildAPL:AddSpell(
ShurikenStorm:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
Player:GetEnemies(10) >= 2 + ((Gloomblade:IsKnown() and
(Player:GetAuras():FindMy(LingeringShadow):GetRemainingTime() >= 6 or
Player:GetAuras():FindMy(PerforatedVeins):IsUp())) and 1 or 0)
end
):SetTarget(Target)
)
-- GetTimeToShurikenTornado
--[[
spec:RegisterStateTable( "time_to_sht", setmetatable( {}, {
__index = function( t, k )
local n = tonumber( k )
n = n - ( n % 1 )
if not n or n > 5 then return 3600 end
if n <= swings_since_sht then return 0 end
local mh_speed = swings.mainhand_speed
local mh_next = ( swings.mainhand > now - 3 ) and ( swings.mainhand + mh_speed ) or now + ( mh_speed * 0.5 )
local oh_speed = swings.offhand_speed
local oh_next = ( swings.offhand > now - 3 ) and ( swings.offhand + oh_speed ) or now
table.wipe( sht )
if mh_speed and mh_speed > 0 then
for i = 1, 4 do
insert( sht, mh_next + ( i * mh_speed ) )
end
end
if oh_speed and oh_speed > 0 then
for i = 1, 4 do
insert( sht, oh_next + ( i * oh_speed ) )
end
end
local i = 1
while( sht[i] ) do
if sht[i] < last_shadow_techniques + 3 then
table.remove( sht, i )
else
i = i + 1
end
end
if #sht > 0 and n - swings_since_sht < #sht then
table.sort( sht )
return max( 0, sht[ n - swings_since_sht ] - query_time )
else
return 3600
end
end
} ) )
]]
function GetTimeToShurikenTornado(unit, n)
local now = GetTime()
local sht = {}
local swings = unit:GetSwingTimers()
if not unit.swings_since_sht then
unit.swings_since_sht = 0
end
if not unit.last_shadow_techniques then
unit.last_shadow_techniques = 0
end
if n <= unit.swings_since_sht then
return 0
end
local mh_speed = swings[1]
local mh_next = (unit.last_mh > now - 3) and (unit.last_mh + mh_speed) or now + (mh_speed * 0.5)
local oh_speed = swings[2]
local oh_next = (unit.last_oh > now - 3) and (unit.last_oh + oh_speed) or now
table.wipe(sht)
if mh_speed and mh_speed > 0 then
for i = 1, 4 do
table.insert(sht, mh_next + (i * mh_speed))
end
end
if oh_speed and oh_speed > 0 then
for i = 1, 4 do
table.insert(sht, oh_next + (i * oh_speed))
end
end
local i = 1
while (sht[i]) do
if sht[i] < unit.last_shadow_techniques + 3 then
table.remove(sht, i)
else
i = i + 1
end
end
if #sht > 0 and n - unit.swings_since_sht < #sht then
table.sort(sht)
return math.max(0, sht[n - unit.swings_since_sht] - now)
else
return 3600
end
end
-- # Build immediately unless the next CP is Animacharged and we won't cap energy waiting for it.
-- actions.build+=/variable,name=anima_helper,value=!talent.echoing_reprimand.enabled|!(variable.is_next_cp_animacharged&(time_to_sht.3.plus<0.5|time_to_sht.4.plus<1)&energy<60)
BuildAPL:AddVariable(
'anima_helper',
function()
return not EchoingReprimand:IsKnown() or ((not (DefaultAPL:GetVariable('is_next_cp_animacharged') and
(GetTimeToShurikenTornado(Player, 3) < 0.5 or GetTimeToShurikenTornado(Player, 4) < 1) and
Player:GetPower() < 60)))
end
)
-- actions.build+=/gloomblade,if=variable.anima_helper
BuildAPL:AddSpell(
Gloomblade:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and BuildAPL:GetVariable('anima_helper')
end
):SetTarget(Target)
)
-- actions.build+=/backstab,if=variable.anima_helper
BuildAPL:AddSpell(
Backstab:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and BuildAPL:GetVariable('anima_helper')
end
):SetTarget(Target)
)
-- # Cooldowns Use Dance off-gcd before the first Shuriken Storm from Tornado comes in.
-- actions.cds=shadow_dance,use_off_gcd=1,if=!buff.shadow_dance.up&buff.shuriken_tornado.up&buff.shuriken_tornado.remains<=3.5
CDsAPL:AddSpell(
ShadowDance:CastableIf(
function(self)
return Player:IsAffectingCombat() and self:IsKnownAndUsable() and
not Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
Player:GetAuras():FindMy(ShurikenTornado):IsUp() and
Player:GetAuras():FindMy(ShurikenTornado):GetRemainingTime() <= 3.5
end
):SetTarget(Player)
)
-- # (Unless already up because we took Shadow Focus) use Symbols off-gcd before the first Shuriken Storm from Tornado comes in.
-- actions.cds+=/symbols_of_death,use_off_gcd=1,if=buff.shuriken_tornado.up&buff.shuriken_tornado.remains<=3.5
CDsAPL:AddSpell(
SymbolsOfDeath:CastableIf(
function(self)
return Player:IsAffectingCombat() and self:IsKnownAndUsable() and
Player:GetAuras():FindMy(ShurikenTornado):IsUp() and
Player:GetAuras():FindMy(ShurikenTornado):GetRemainingTime() <= 3.5
end
):SetTarget(Player)
)
-- # Vanish for Shadowstrike with Danse Macabre at adaquate stacks
-- actions.cds+=/vanish,if=buff.danse_macabre.stack>3&combo_points<=2
CDsAPL:AddSpell(
Vanish:CastableIf(
function(self)
return Player:IsAffectingCombat() and Player:GetAuras():FindMy(Vanish):IsUp() and self:IsKnownAndUsable() and
Player:GetAuras():FindMy(DanseMacabre):GetCount() > 3 and
DefaultAPL:GetVariable('effective_combo_points') <= 2 and Vanish:GetTimeSinceLastCast() > 2
end
):SetTarget(Player)
)
-- # Cold Blood on 5 combo points when not playing Secret Technique
-- actions.cds+=/cold_blood,if=!talent.secret_technique&combo_points>=5
CDsAPL:AddSpell(
ColdBlood:CastableIf(
function(self)
return self:IsKnownAndUsable() and not SecretTechnique:IsKnown() and
DefaultAPL:GetVariable('effective_combo_points') >= 5
end
):SetTarget(Player)
)
-- actions.cds+=/flagellation,target_if=max:target.time_to_die,if=variable.snd_condition&combo_points>=5&target.time_to_die>10
CDsAPL:AddSpell(
Flagellation:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and DefaultAPL:GetVariable('snd_condition') and
DefaultAPL:GetVariable('effective_combo_points') >= 5 and
Target:TimeToDie() > 10
end
):SetTarget(Target)
)
-- # Pool for Tornado pre-SoD with ShD ready when not running SF.
-- actions.cds+=/pool_resource,for_next=1,if=talent.shuriken_tornado.enabled&!talent.shadow_focus.enabled
-- # Use Tornado pre SoD when we have the energy whether from pooling without SF or just generally.
-- actions.cds+=/shuriken_tornado,if=spell_targets.shuriken_storm<=1&energy>=60&variable.snd_condition&cooldown.symbols_of_death.up&cooldown.shadow_dance.charges>=1&(!talent.flagellation.enabled&!cooldown.flagellation.up|buff.flagellation_buff.up|spell_targets.shuriken_storm>=5)&combo_points<=2&!buff.premeditation.up
CDsAPL:AddSpell(
ShurikenTornado:CastableIf(
function(self)
return self:IsKnownAndUsable() and Player:GetEnemies(10) <= 1 and
Player:GetPower() >= 60 and
DefaultAPL:GetVariable('snd_condition') and
SymbolsOfDeath:OnCooldown() and
ShadowDance:GetCharges() >= 1 and
(not Flagellation:IsKnown() and not Flagellation:OnCooldown() or
Player:GetAuras():FindMy(Flagellation):IsUp() or
Player:GetEnemies(10) >= 5) and
DefaultAPL:GetVariable('effective_combo_points') <= 2 and
not Player:GetAuras():FindMy(Premeditation):IsUp()
end
):SetTarget(Target)
)
-- actions.cds+=/sepsis,if=variable.snd_condition&combo_points.deficit>=1&target.time_to_die>=16
CDsAPL:AddSpell(
Sepsis:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and DefaultAPL:GetVariable('snd_condition') and
Player:GetComboPointsDeficit() >= 1 and
Target:TimeToDie() >= 16
end
):SetTarget(Target)
)
-- # Use Symbols on cooldown (after first SnD) unless we are going to pop Tornado and do not have Shadow Focus.
-- actions.cds+=/symbols_of_death,if=variable.snd_condition&(!talent.flagellation|cooldown.flagellation.remains>10|cooldown.flagellation.up&combo_points>=5)
CDsAPL:AddSpell(
SymbolsOfDeath:CastableIf(
function(self)
return self:IsKnownAndUsable() and DefaultAPL:GetVariable('snd_condition') and
(not Flagellation:IsKnown() or
Flagellation:GetCooldownRemaining() > 10 or
Flagellation:OnCooldown() and DefaultAPL:GetVariable('effective_combo_points') >= 5)
end
):SetTarget(Player)
)
-- # If adds are up, snipe the one with lowest TTD. Use when dying faster than CP deficit or not stealthed without any CP.
-- actions.cds+=/marked_for_death,line_cd=1.5,target_if=min:target.time_to_die,if=raid_event.adds.up&(target.time_to_die<combo_points.deficit|!stealthed.all&combo_points.deficit>=cp_max_spend)
-- # If no adds will die within the next 30s, use MfD on boss without any CP.
-- actions.cds+=/marked_for_death,if=raid_event.adds.in>30-raid_event.adds.duration&combo_points.deficit>=cp_max_spend
-- actions.cds+=/shadow_blades,if=variable.snd_condition&combo_points.deficit>=2&target.time_to_die>=10&(dot.sepsis.ticking|cooldown.sepsis.remains<=8|!talent.sepsis)|fight_remains<=20
CDsAPL:AddSpell(
ShadowBlades:CastableIf(
function(self)
return Player:IsAffectingCombat() and self:IsKnownAndUsable() and
DefaultAPL:GetVariable('snd_condition') and
Player:GetComboPointsDeficit() >= 2 and
Target:TimeToDie() >= 10 and
((Target:GetAuras():FindMy(Sepsis):IsUp() or
Sepsis:GetCooldownRemaining() <= 8 or
not Sepsis:IsKnown()) or
Target:TimeToDie() <= 20)
end
):SetTarget(Target)
)
-- actions.cds+=/echoing_reprimand,if=variable.snd_condition&combo_points.deficit>=3&(variable.priority_rotation|spell_targets.shuriken_storm<=4|talent.resounding_clarity)&(buff.shadow_dance.up|!talent.danse_macabre)
CDsAPL:AddSpell(
EchoingReprimand:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and DefaultAPL:GetVariable('snd_condition') and
Player:GetComboPointsDeficit() >= 3 and
(DefaultAPL:GetVariable('priority_rotation') or
Player:GetEnemies(10) <= 4 or
ResoundingClarity:IsKnown()) and
(Player:GetAuras():FindMy(ShadowDance):IsUp() or
not DanseMacabre:IsKnown())
end
):SetTarget(Target)
)
-- # With SF, if not already done, use Tornado with SoD up.
-- actions.cds+=/shuriken_tornado,if=variable.snd_condition&buff.symbols_of_death.up&combo_points<=2&(!buff.premeditation.up|spell_targets.shuriken_storm>4)
CDsAPL:AddSpell(
ShurikenTornado:CastableIf(
function(self)
return self:IsKnownAndUsable() and DefaultAPL:GetVariable('snd_condition') and
Player:GetAuras():FindMy(SymbolsOfDeath):IsUp() and
DefaultAPL:GetVariable('effective_combo_points') <= 2 and
(not Player:GetAuras():FindMy(Premeditation):IsUp() or
Player:GetEnemies(10) > 4)
end
):SetTarget(Target)
)
-- actions.cds+=/shuriken_tornado,if=cooldown.shadow_dance.ready&!stealthed.all&spell_targets.shuriken_storm>=3&!talent.flagellation.enabled
CDsAPL:AddSpell(
ShurikenTornado:CastableIf(
function(self)
return self:IsKnownAndUsable() and ShadowDance:GetCooldownRemaining() == 0 and
not Player:IsStealthed() and
Player:GetEnemies(10) >= 3 and
not Flagellation:IsKnown()
end
):SetTarget(Target)
)
-- actions.cds+=/shadow_dance,if=!buff.shadow_dance.up&fight_remains<=8+talent.subterfuge.enabled
CDsAPL:AddSpell(
ShadowDance:CastableIf(
function(self)
return Player:IsAffectingCombat() and self:IsKnownAndUsable() and
not Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
Target:TimeToDie() <= 8 + (Subterfuge:IsKnown() and 1 or 0)
end
):SetTarget(Target)
)
-- actions.cds+=/thistle_tea,if=cooldown.symbols_of_death.remains>=3&!buff.thistle_tea.up&(energy.deficit>=100|cooldown.thistle_tea.charges_fractional>=2.75&buff.shadow_dance.up)|buff.shadow_dance.remains>=4&!buff.thistle_tea.up&spell_targets.shuriken_storm>=3|!buff.thistle_tea.up&fight_remains<=(6*cooldown.thistle_tea.charges)
CDsAPL:AddSpell(
ThistleTea:CastableIf(
function(self)
return Player:IsAffectingCombat() and self:IsKnownAndUsable() and
(SymbolsOfDeath:GetCooldownRemaining() >= 3
and not Player:GetAuras():FindMy(ThistleTea):IsUp() and
(Player:GetPowerDeficit() >= 100 or
self:GetChargesFractional() >= 2.75 and
Player:GetAuras():FindMy(ShadowDance):IsUp()) or
(Player:GetAuras():FindMy(ShadowDance):GetRemainingTime() >= 4 and
not Player:GetAuras():FindMy(ThistleTea):IsUp() and
Player:GetEnemies(10) >= 3) or
(not Player:GetAuras():FindMy(ThistleTea):IsUp() and
Player:TimeToDie() <= (6 * self:GetCharges())))
end
):SetTarget(Player)
)
-- actions.cds+=/potion,if=buff.bloodlust.react|fight_remains<30|buff.symbols_of_death.up&(buff.shadow_blades.up|cooldown.shadow_blades.remains<=10)
-- actions.cds+=/blood_fury,if=buff.symbols_of_death.up
-- actions.cds+=/berserking,if=buff.symbols_of_death.up
-- actions.cds+=/fireblood,if=buff.symbols_of_death.up
-- actions.cds+=/ancestral_call,if=buff.symbols_of_death.up
-- actions.cds+=/use_item,name=manic_grieftorch,use_off_gcd=1,if=gcd.remains>gcd.max-0.1,if=!stealthed.all
-- # Default fallback for usable items: Use with Symbols of Death.
-- actions.cds+=/use_items,if=buff.symbols_of_death.up|fight_remains<20
CDsAPL:AddAPL(
ItemsAPL,
function()
return Player:GetAuras():FindMy(SymbolsOfDeath):IsUp() or (Target:IsBoss() and Target:TimeToDie() < 20)
end
)
-- # Finishers While using Premeditation, avoid casting Slice and Dice when Shadow Dance is soon to be used, except for Kyrian
-- actions.finish=variable,name=premed_snd_condition,value=talent.premeditation.enabled&spell_targets.shuriken_storm<5
FinishAPL:AddVariable(
'premed_snd_condition',
function()
return Premeditation:IsKnown() and Player:GetEnemies(10) < 5
end
)
-- return ( cp + 1 ) * p()->buffs.slice_and_dice->data().duration();
function GetTriggeredDuration()
return (Player:GetComboPoints() + 1) * 6
end
-- return p()->buffs.slice_and_dice->remains() < get_triggered_duration( as<int>( p()->current_effective_cp( false ) ) ) * 0.3;
function Refreshable(aura, target)
return target:GetAuras():FindMy(aura):GetRemainingTime() < GetTriggeredDuration() * 0.3
end
-- actions.finish+=/slice_and_dice,if=!variable.premed_snd_condition&spell_targets.shuriken_storm<6&!buff.shadow_dance.up&buff.slice_and_dice.remains<fight_remains&refreshable
FinishAPL:AddSpell(
SliceAndDice:CastableIf(
function(self)
return self:IsKnownAndUsable() and not DefaultAPL:GetVariable('premed_snd_condition') and
Player:GetEnemies(10) < 6 and
not Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
Player:GetAuras():FindMy(SliceAndDice):GetRemainingTime() < Target:TimeToDie() and
Refreshable(SliceAndDice, Player)
end
):SetTarget(Player)
)
-- actions.finish+=/slice_and_dice,if=variable.premed_snd_condition&cooldown.shadow_dance.charges_fractional<1.75&buff.slice_and_dice.remains<cooldown.symbols_of_death.remains&(cooldown.shadow_dance.ready&buff.symbols_of_death.remains-buff.shadow_dance.remains<1.2)
FinishAPL:AddSpell(
SliceAndDice:CastableIf(
function(self)
return self:IsKnownAndUsable() and DefaultAPL:GetVariable('premed_snd_condition') and
ShadowDance:GetChargesFractional() < 1.75 and
Player:GetAuras():FindMy(SliceAndDice):GetRemainingTime() < SymbolsOfDeath:GetCooldownRemaining() and
(ShadowDance:OnCooldown() and
Player:GetAuras():FindMy(SymbolsOfDeath):GetRemainingTime() -
Player:GetAuras():FindMy(ShadowDanceAura):GetRemainingTime() < 1.2)
end
):SetTarget(Player)
)
-- actions.finish+=/variable,name=skip_rupture,value=buff.thistle_tea.up&spell_targets.shuriken_storm=1|buff.shadow_dance.up&(spell_targets.shuriken_storm=1|dot.rupture.ticking&spell_targets.shuriken_storm>=2)
FinishAPL:AddVariable(
'skip_rupture',
function()
return (Player:GetAuras():FindMy(ThistleTea):IsUp() and Player:GetEnemies(10) == 1) or
(Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
(Player:GetEnemies(10) == 1 or
Target:GetAuras():FindMy(Rupture):IsUp() and Player:GetEnemies(10) >= 2))
end
)
-- # Keep up Rupture if it is about to run out.
-- actions.finish+=/rupture,if=(!variable.skip_rupture|variable.priority_rotation)&target.time_to_die-remains>6&refreshable
FinishAPL:AddSpell(
Rupture:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
(not DefaultAPL:GetVariable('skip_rupture') or
DefaultAPL:GetVariable('priority_rotation')) and
Target:TimeToDie() - Target:GetAuras():FindMy(Rupture):GetRemainingTime() > 6 and
Refreshable(Rupture, Target)
end
):SetTarget(Target)
)
-- # Refresh Rupture early for Finality
-- actions.finish+=/rupture,if=!variable.skip_rupture&buff.finality_rupture.up&cooldown.shadow_dance.remains<12&cooldown.shadow_dance.charges_fractional<=1&spell_targets.shuriken_storm=1&(talent.dark_brew|talent.danse_macabre)
FinishAPL:AddSpell(
Rupture:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and not DefaultAPL:GetVariable('skip_rupture') and
Player:GetAuras():FindMy(FinalityRupture):IsUp() and
ShadowDance:GetCooldownRemaining() < 12 and
ShadowDance:GetChargesFractional() <= 1 and
Player:GetEnemies(10) == 1 and
(DarkBrew:IsKnown() or DanseMacabre:IsKnown())
end
):SetTarget(Target)
)
-- # Sync Cold Blood with Secret Technique when possible
-- actions.finish+=/cold_blood,if=buff.shadow_dance.up&(buff.danse_macabre.stack>=3|!talent.danse_macabre)&cooldown.secret_technique.ready
FinishAPL:AddSpell(
ColdBlood:CastableIf(
function(self)
return self:IsKnownAndUsable() and Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
(Player:GetAuras():FindMy(DanseMacabre):GetCount() >= 3 or
not DanseMacabre:IsKnown()) and
SecretTechnique:OnCooldown()
end
):SetTarget(Target)
)
-- actions.finish+=/secret_technique,if=buff.shadow_dance.up&(buff.danse_macabre.stack>=3|!talent.danse_macabre)&(!talent.cold_blood|cooldown.cold_blood.remains>buff.shadow_dance.remains-2)
FinishAPL:AddSpell(
SecretTechnique:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
Player:GetAuras():FindMy(ShadowDanceAura):IsUp() and
(Player:GetAuras():FindMy(DanseMacabre):GetCount() >= 3 or
not DanseMacabre:IsKnown()) and
(not ColdBlood:IsKnown() or
ColdBlood:GetCooldownRemaining() > Player:GetAuras():FindMy(ShadowDanceAura):GetRemainingTime() - 2)
end
):SetTarget(Target)
)
-- # Multidotting targets that will live for the duration of Rupture, refresh during pandemic.
-- actions.finish+=/rupture,cycle_targets=1,if=!variable.skip_rupture&!variable.priority_rotation&spell_targets.shuriken_storm>=2&target.time_to_die>=(2*combo_points)&refreshable
FinishAPL:AddSpell(
Rupture:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(RuptureTarget) and
not DefaultAPL:GetVariable('skip_rupture') and
not DefaultAPL:GetVariable('priority_rotation') and
Player:GetEnemies(10) >= 2 and
RuptureTarget:TimeToDie() >= (2 * Player:GetComboPoints()) and
Refreshable(Rupture, RuptureTarget)
end
):SetTarget(RuptureTarget)
)
-- # Refresh Rupture early if it will expire during Symbols. Do that refresh if SoD gets ready in the next 5s.
-- actions.finish+=/rupture,if=!variable.skip_rupture&remains<cooldown.symbols_of_death.remains+10&cooldown.symbols_of_death.remains<=5&target.time_to_die-remains>cooldown.symbols_of_death.remains+5
FinishAPL:AddSpell(
Rupture:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and not DefaultAPL:GetVariable('skip_rupture') and
Target:GetAuras():FindMy(Rupture):GetRemainingTime() <
SymbolsOfDeath:GetCooldownRemaining() + 10 and
SymbolsOfDeath:GetCooldownRemaining() <= 5 and
Target:TimeToDie() - Target:GetAuras():FindMy(Rupture):GetRemainingTime() >
SymbolsOfDeath:GetCooldownRemaining() + 5
end
):SetTarget(Target)
)
-- actions.finish+=/black_powder,if=!variable.priority_rotation&spell_targets>=3
FinishAPL:AddSpell(
BlackPowder:CastableIf(
function(self)
return self:IsKnownAndUsable() and not DefaultAPL:GetVariable('priority_rotation') and
Player:GetEnemies(10) >= 3
end
):SetTarget(Target)
)
-- actions.finish+=/eviscerate
FinishAPL:AddSpell(
Eviscerate:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and true
end
):SetTarget(Target)
)
-- # Stealth Cooldowns Helper Variable
-- actions.stealth_cds=variable,name=shd_threshold,value=cooldown.shadow_dance.charges_fractional>=0.75+talent.shadow_dance
StealthCDsAPL:AddVariable(
'shd_threshold',
function(self)
return ShadowDance:GetChargesFractional() >= 0.75 + (ShadowDance:IsKnown() and 1 or 0)
end
)
-- # Vanish if we are capping on Dance charges. Early before first dance if we have no Nightstalker but Dark Shadow in order to get Rupture up (no Master Assassin).
-- actions.stealth_cds+=/vanish,if=(!talent.danse_macabre|spell_targets.shuriken_storm>=3)&!variable.shd_threshold&combo_points.deficit>1
StealthCDsAPL:AddSpell(
Vanish:CastableIf(
function(self)
return Player:IsAffectingCombat() and not Player:GetAuras():FindMy(Stealth):IsUp() and
self:IsKnownAndUsable() and
(not DanseMacabre:IsKnown() or Player:GetEnemies(10) >= 3) and
not StealthCDsAPL:GetVariable('shd_threshold') and
Player:GetComboPointsDeficit() > 1 and Vanish:GetTimeSinceLastCast() > 2
end
):SetTarget(Target)
)
-- # Pool for Shadowmeld + Shadowstrike unless we are about to cap on Dance charges. Only when Find Weakness is about to run out.
-- actions.stealth_cds+=/pool_resource,for_next=1,extra_amount=40,if=race.night_elf
-- actions.stealth_cds+=/shadowmeld,if=energy>=40&energy.deficit>=10&!variable.shd_threshold&combo_points.deficit>4
StealthCDsAPL:AddSpell(
Shadowmeld:CastableIf(
function(self)
return self:IsKnownAndUsable() and Player:GetEnergy() >= 40 and
Player:GetPowerDeficit() >= 10 and
not StealthCDsAPL:GetVariable('shd_threshold') and
Player:GetComboPointsDeficit() > 4
end
):SetTarget(Target)
)
-- # CP thresholds for entering Shadow Dance Default to start dance with 0 or 1 combo point
-- actions.stealth_cds+=/variable,name=shd_combo_points,value=combo_points<=1
StealthCDsAPL:AddVariable(
'shd_combo_points',
function(self)
return Player:GetComboPoints() <= 1
end
)
-- # Use stealth cooldowns with high combo points when playing shuriken tornado or with high target counts
-- actions.stealth_cds+=/variable,name=shd_combo_points,value=combo_points.deficit<=1,if=spell_targets.shuriken_storm>(4-2*talent.shuriken_tornado.enabled)|variable.priority_rotation&spell_targets.shuriken_storm>=4
StealthCDsAPL:AddVariable(
'shd_combo_points',
function(self)
return Player:GetComboPointsDeficit() <= 1 and
((Player:GetEnemies(10) > (4 - 2 * (ShurikenTornado:IsKnown() and 1 or 0))) or
(DefaultAPL:GetVariable('priority_rotation') and
Player:GetEnemies(10) >= 4))
end
)
-- # Use stealth cooldowns on any combo point on 4 targets
-- actions.stealth_cds+=/variable,name=shd_combo_points,value=1,if=spell_targets.shuriken_storm=(4-talent.seal_fate)
StealthCDsAPL:AddVariable(
'shd_combo_points',
function(self)
return Player:GetEnemies(10) == (4 - (SealFate:IsKnown() and 1 or 0))
end
)
-- # Dance during Symbols or above threshold.
-- actions.stealth_cds+=/shadow_dance,if=(variable.shd_combo_points&(buff.symbols_of_death.remains>=(2.2-talent.flagellation.enabled)|variable.shd_threshold)|buff.flagellation.up|buff.flagellation_persist.remains>=6|spell_targets.shuriken_storm>=4&cooldown.symbols_of_death.remains>10)&!buff.the_rotten.up
StealthCDsAPL:AddSpell(
ShadowDance:CastableIf(
function(self)
return Player:IsAffectingCombat() and self:IsKnownAndUsable() and
((StealthCDsAPL:GetVariable('shd_combo_points') and
(Player:GetAuras():FindMy(SymbolsOfDeath):GetRemainingTime() >=
(2.2 - (Flagellation:IsKnown() and 1 or 0)) or
StealthCDsAPL:GetVariable('shd_threshold'))) or
Player:GetAuras():FindMy(Flagellation):IsUp() or
Player:GetAuras():FindMy(FlagellationPersist):GetRemainingTime() >= 6 or
Player:GetEnemies(10) >= 4 and
SymbolsOfDeath:GetCooldownRemaining() > 10) and
not Target:GetAuras():FindMy(TheRotten):IsUp()
end
):SetTarget(Target)
)
-- # Burn Dances charges if before the fight ends if SoD won't be ready in time.
-- actions.stealth_cds+=/shadow_dance,if=variable.shd_combo_points&fight_remains<cooldown.symbols_of_death.remains|!talent.shadow_dance&dot.rupture.ticking&spell_targets.shuriken_storm<=4&!buff.the_rotten.up
StealthCDsAPL:AddSpell(
ShadowDance:CastableIf(
function(self)
return Player:IsAffectingCombat() and self:IsKnownAndUsable() and
((StealthCDsAPL:GetVariable('shd_combo_points') and
Target:TimeToDie() < SymbolsOfDeath:GetCooldownRemaining()) or
(
Target:GetAuras():FindMy(Rupture):IsUp() and
Player:GetEnemies(10) <= 4 and
not Target:GetAuras():FindMy(TheRotten):IsUp()))
end
):SetTarget(Target)
)
-- # Stealthed Rotation If Stealth/vanish are up, use Shadowstrike to benefit from the passive bonus and Find Weakness, even if we are at max CP (unless using Master Assassin)
-- actions.stealthed=shadowstrike,if=(buff.stealth.up|buff.vanish.up)&(spell_targets.shuriken_storm<4|variable.priority_rotation)
StealthedAPL:AddSpell(
Shadowstrike:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and (Player:GetAuras():FindMy(Stealth):IsUp() or
Player:GetAuras():FindMy(Vanish):IsUp()) and
(Player:GetEnemies(10) < 4 or
DefaultAPL:GetVariable('priority_rotation'))
end
):SetTarget(Target)
)
-- # Variable to Gloomblade / Backstab when on 4 or 5 combo points with premediation and when the combo point is not anima charged
-- actions.stealthed+=/variable,name=gloomblade_condition,value=buff.danse_macabre.stack<5&(combo_points.deficit=2|combo_points.deficit=3)&(buff.premeditation.up|effective_combo_points<7)&(spell_targets.shuriken_storm<=8|talent.lingering_shadow)
StealthedAPL:AddVariable(
'gloomblade_condition',
function(self)
return Player:GetAuras():FindMy(DanseMacabre):GetCount() < 5 and
(Player:GetComboPointsDeficit() == 2 or
Player:GetComboPointsDeficit() == 3) and
(Player:GetAuras():FindMy(Premeditation):IsUp() or
DefaultAPL:GetVariable('effective_combo_points') < 7) and
(Player:GetEnemies(10) <= 8 or
LingeringShadow:IsKnown())
end
)
-- actions.stealthed+=/shuriken_storm,if=variable.gloomblade_condition&buff.silent_storm.up&!debuff.find_weakness.remains&talent.improved_shuriken_storm.enabled
StealthedAPL:AddSpell(
ShurikenStorm:CastableIf(
function(self)
return self:IsKnownAndUsable() and StealthedAPL:GetVariable('gloomblade_condition') and
Player:GetAuras():FindMy(SilentStorm):IsUp() and
not Target:GetAuras():FindMy(FindWeakness):GetRemainingTime() and
ImprovedShurikenStorm:IsKnown()
end
):SetTarget(Target)
)
-- actions.stealthed+=/gloomblade,if=variable.gloomblade_condition
StealthedAPL:AddSpell(
Gloomblade:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
StealthedAPL:GetVariable('gloomblade_condition')
end
):SetTarget(Target)
)
-- actions.stealthed+=/backstab,if=variable.gloomblade_condition&talent.danse_macabre&buff.danse_macabre.stack<=2&spell_targets.shuriken_storm<=2
StealthedAPL:AddSpell(
Backstab:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
StealthedAPL:GetVariable('gloomblade_condition') and
DanseMacabre:IsKnown() and
Player:GetAuras():FindMy(DanseMacabre):GetCount() <= 2 and
Player:GetEnemies(10) <= 2
end
):SetTarget(Target)
)
--[[
double consume_cp_max() const
{
return COMBO_POINT_MAX + as<double>( talent.rogue.deeper_stratagem->effectN( 2 ).base_value() +
talent.outlaw.devious_stratagem->effectN( 2 ).base_value() +
talent.subtlety.secret_stratagem->effectN( 2 ).base_value() );
}
]]
function ConsumeCPMax()
return 5 + (DeeperStratagem:IsKnownAndUsable() and 1 or 0) + (DeviousStratagem:IsKnownAndUsable() and 1 or 0) +
(SecretStratagem:IsKnownAndUsable() and 1 or 0)
end
-- actions.stealthed+=/call_action_list,name=finish,if=variable.effective_combo_points>=cp_max_spend
StealthedAPL:AddAPL(
FinishAPL,
function(self)
return DefaultAPL:GetVariable('effective_combo_points') >= ConsumeCPMax()
end
)
-- # Finish earlier with Shuriken tornado up.
-- actions.stealthed+=/call_action_list,name=finish,if=buff.shuriken_tornado.up&combo_points.deficit<=2
StealthedAPL:AddAPL(
FinishAPL,
function(self)
return Player:GetAuras():FindMy(ShurikenTornado):IsUp() and
Player:GetComboPointsDeficit() <= 2
end
)
-- # Also safe to finish at 4+ CP with exactly 4 targets. (Same as outside stealth.)
-- actions.stealthed+=/call_action_list,name=finish,if=spell_targets.shuriken_storm>=4-talent.seal_fate&variable.effective_combo_points>=4
StealthedAPL:AddAPL(
FinishAPL,
function()
return Player:GetEnemies(10) >= 4 - ((SealFate:IsKnown() and
DefaultAPL:GetVariable('effective_combo_points') >= 4) and 1 or 0)
end
)
-- # Finish at lower combo points if you are talented in DS, SS or Seal Fate
-- actions.stealthed+=/call_action_list,name=finish,if=combo_points.deficit<=1+(talent.seal_fate|talent.deeper_stratagem|talent.secret_stratagem)
StealthedAPL:AddAPL(
FinishAPL,
function()
return Player:GetComboPointsDeficit() <=
1 + ((SealFate:IsKnown() or DeeperStratagem:IsKnown() or SecretStratagem:IsKnown()) and 1 or 0)
end
)
-- # Use Gloomblade or Backstab when close to hitting max PV stacks
-- actions.stealthed+=/gloomblade,if=buff.perforated_veins.stack>=5&spell_targets.shuriken_storm<3
StealthedAPL:AddSpell(
Gloomblade:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
Player:GetAuras():FindMy(PerforatedVeins):GetCount() >= 5 and
Player:GetEnemies(10) < 3
end
):SetTarget(Target)
)
-- actions.stealthed+=/backstab,if=buff.perforated_veins.stack>=5&spell_targets.shuriken_storm<3
StealthedAPL:AddSpell(
Backstab:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
Player:GetAuras():FindMy(PerforatedVeins):GetCount() >= 5 and
Player:GetEnemies(10) < 3
end
):SetTarget(Target)
)
-- actions.stealthed+=/shadowstrike,if=stealthed.sepsis&spell_targets.shuriken_storm<4
StealthedAPL:AddSpell(
Shadowstrike:CastableIf(
function(self)
return self:IsKnownAndUsable() and Player:GetAuras():FindMy(Sepsis):IsUp() and
Player:GetEnemies(10) < 4
end
):SetTarget(Target)
)
-- actions.stealthed+=/shuriken_storm,if=spell_targets>=3+buff.the_rotten.up&(!buff.premeditation.up|spell_targets>=7)
StealthedAPL:AddSpell(
ShurikenStorm:CastableIf(
function(self)
return self:IsKnownAndUsable() and
Player:GetEnemies(10) >= 3 + ((Player:GetAuras():FindMy(TheRotten):IsUp() and
(not Player:GetAuras():FindMy(Premeditation):IsUp() or Player:GetEnemies(10) >= 7)) and 1 or 0)
end
):SetTarget(Target)
)
-- # Shadowstrike to refresh Find Weakness and to ensure we can carry over a full FW into the next SoD if possible.
-- actions.stealthed+=/shadowstrike,if=debuff.find_weakness.remains<=1|cooldown.symbols_of_death.remains<18&debuff.find_weakness.remains<cooldown.symbols_of_death.remains
StealthedAPL:AddSpell(
Shadowstrike:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target) and
(Target:GetAuras():FindMy(FindWeakness):GetRemainingTime() <= 1 or
(SymbolsOfDeath:GetCooldownRemaining() < 18 and
Target:GetAuras():FindMy(FindWeakness):GetRemainingTime() < SymbolsOfDeath:GetCooldownRemaining()
))
end
):SetTarget(Target)
)
-- actions.stealthed+=/shadowstrike
StealthedAPL:AddSpell(
Shadowstrike:CastableIf(
function(self)
return self:IsKnownAndUsable() and self:IsInRange(Target)
end
):SetTarget(Target)
)
Stealth:CastableIf(
function(self)
return self:IsKnownAndUsable() and not Player:GetAuras():FindMy(Stealth):IsUp() and
not Player:IsAffectingCombat() and not IsMounted()
end
)
SubModulue:Sync(function()
DefaultAPL:Execute()
end)
Bastion:Register(SubModulue)