Untitled
function(event, ...) local aura_env = aura_env -- Prevent code execution during loading screens or WA options window if event == "LOADING_SCREEN_DISABLED" then aura_env.world_loaded = true elseif event == "LOADING_SCREEN_ENABLED" then aura_env.world_loaded = false return false elseif event == "OPTIONS" or not aura_env.world_loaded then return false end local _GetTime = GetTime local GetTime = function() return aura_env.frameTime or _GetTime() end if event == "FRAME_UPDATE" then aura_env.frameTime = _GetTime() end -- This should never occur if not aura_env.CPlayer then return false end -- Cache Globals local Player = aura_env.CPlayer local Combat = Player.combat local spec = Player.spec local bit = bit local bit_band = bit.band local C_NamePlate = C_NamePlate local GetNamePlateForUnit = C_NamePlate.GetNamePlateForUnit local GetNamePlates = C_NamePlate.GetNamePlates local C_Scenario = C_Scenario local GetCriteriaInfo = C_Scenario.GetCriteriaInfo or C_ScenarioInfo.GetCriteriaInfo local GetInfo = C_Scenario.GetInfo local GetStepInfo = C_Scenario.GetStepInfo local C_UnitAuras = C_UnitAuras local GetAuraDataByAuraInstanceID = C_UnitAuras.GetAuraDataByAuraInstanceID local GetAuraDataBySlot = C_UnitAuras.GetAuraDataBySlot local CheckInteractDistance = CheckInteractDistance local COMBATLOG_OBJECT_AFFILIATION_MASK = COMBATLOG_OBJECT_AFFILIATION_MASK local COMBATLOG_OBJECT_AFFILIATION_MINE = COMBATLOG_OBJECT_AFFILIATION_MINE local CombatLogGetCurrentEventInfo = CombatLogGetCurrentEventInfo local debugprofilestart = debugprofilestart local debugprofilestop = debugprofilestop local floor = floor local frameTime = GetTime() local GetPowerRegen = GetPowerRegen local GetRaidTargetIndex = GetRaidTargetIndex local GetSpellCharges = GetSpellCharges or function( spellID ) if not spellID then return nil end local chargeInfo = C_Spell.GetSpellCharges( spellID ) if chargeInfo then return chargeInfo.currentCharges, chargeInfo.maxCharges, chargeInfo.cooldownStartTime, chargeInfo.cooldownDuration, chargeInfo.chargeModRate end end local GetSpellCooldown = GetSpellCooldown or function( spellID ) local info = C_Spell.GetSpellCooldown( spellID ) return info.startTime, info.duration, info.isEnabled end local GetSpellPowerCost = GetSpellPowerCost or C_Spell.GetSpellPowerCost local GetUnitSpeed = GetUnitSpeed local global_modifier = aura_env.global_modifier local InCombatLockdown = InCombatLockdown local ipairs = ipairs local IsEquippedItemType = IsEquippedItemType or C_Item.IsEquippedItemType local IsMounted = IsMounted local IsSpellKnown = IsSpellKnown local LE_SCENARIO_TYPE_CHALLENGE_MODE = LE_SCENARIO_TYPE_CHALLENGE_MODE local LibStub = LibStub local math = math local max = math.max local min = math.min local pairs = pairs local print = print local select = select local SetRaidTarget = SetRaidTarget local string = string local find = string.find local gsub = string.gsub local len = string.len local lower = string.lower local sub = string.sub local table = table local targetAuraEffect = aura_env.targetAuraEffect local tonumber = tonumber local tostring = tostring local type = type local UnitAffectingCombat = UnitAffectingCombat local UnitAuraSlots = UnitAuraSlots or C_UnitAuras.GetAuraSlots local UnitChannelInfo = UnitChannelInfo local UnitExists = UnitExists local UnitGetTotalAbsorbs = UnitGetTotalAbsorbs local UnitGetTotalHealAbsorbs = UnitGetTotalHealAbsorbs local UnitGUID = UnitGUID local UnitHealth = UnitHealth local UnitHealthMax = UnitHealthMax local UnitIsDead = UnitIsDead local UnitIsPlayer = UnitIsPlayer local UnitLevel = UnitLevel local UnitName = UnitName local UnitPower = UnitPower local UnitPowerMax = UnitPowerMax local UnitStagger = UnitStagger local UnitStat = UnitStat local UnitThreatSituation = UnitThreatSituation local WA_IterateGroupMembers = WA_IterateGroupMembers local WeakAuras = WeakAuras local gcdDuration = WeakAuras.gcdDuration local ScanEvents = WeakAuras.ScanEvents local insert = function ( t, v ) t[ #t + 1 ] = v end local next = next local t_sort = table.sort local sort = function( t, func ) if not func then t_sort( t ) else t_sort( t, function( l, r ) if not l then return false elseif not r then return true else return func( l, r ) end end ) end end local profiler_enabled = false aura_env.profiler_out = ( profiler_enabled and aura_env.profiler_out ) or {} local profilerStart = function() if profiler_enabled then debugprofilestart() end end local profilerEnd = function( desc ) if profiler_enabled then local profiler_t = debugprofilestop() if profiler_t and profiler_t > 0 then aura_env.profiler_out[ desc ] = profiler_t end end end -- Main worker thread if event == "FRAME_UPDATE" then local fullUpdate = false -- Development Debugging Option if profiler_enabled and next( aura_env.profiler_out ) ~= nil then ScanEvents( "JEREMYUI_PROFILER", aura_env.profiler_out ) end -- Modulate update speed based on framerate if aura_env.frame_pace then local delta = frameTime - aura_env.frame_pace local modulate = max( 0.0167, delta ) / 0.0167 aura_env.update_rate = min( 1, modulate * 0.5 ) end aura_env.frame_pace = frameTime if not aura_env.RECENT_UPDATE or aura_env.RECENT_UPDATE < frameTime - 0.5 then profilerStart() aura_env.RECENT_UPDATE = frameTime local recentDamage = aura_env.danger_a + aura_env.danger_b for index, damage in pairs( Combat.recent_damage ) do local expire = damage.expire if not expire or expire <= frameTime then Combat.recent_damage[ index ] = nil else recentDamage = recentDamage + ( damage.amount or 0 ) end end -- Used for Touch of Karma evaluation Player.recent_dtps = recentDamage / aura_env.RECENT_DURATION profilerEnd( "Parse Recent Damage" ) end if not aura_env.last or aura_env.fast or aura_env.tick_update and aura_env.tick_update < frameTime or aura_env.last < frameTime - aura_env.update_rate then -- Movement Rate / Speed if aura_env.last then local cur_speed = GetUnitSpeed( "player" ) if InCombatLockdown() and not IsMounted() and cur_speed > 0 then local delta_time = ( frameTime - aura_env.last ) Player.movement_yds = Player.movement_yds + ( cur_speed * delta_time ) Player.movement_t = Player.movement_t + delta_time Player.movement_rate = Player.movement_yds / Player.movement_t else Player.movement_yds = 0 Player.movement_t = 0 Player.movement_rate = 0 end Player.moving = Player.movement_t >= 1.25 end aura_env.fast = false aura_env.last = frameTime fullUpdate = true end aura_env.LBG = aura_env.LBG or LibStub( "LibCustomGlow-1.0" ) local LBG = aura_env.LBG local jeremy = aura_env.jeremy_update or {} local compression_value = 1000 local spellRaw = function ( name ) local raw = jeremy.raw[ name ] or 0 raw = raw * compression_value return raw end jeremy.raw = jeremy.raw or {} jeremy.rank = {} jeremy.scale = {} local spells = aura_env.spells local actionlist = {} local actionhash = {} local default_action = Player.default_action local rate_time = function() return ( aura_env.base_gm and Player.action_modifier / aura_env.base_gm ) or 1 end aura_env.action_cache = aura_env.action_cache or {} local action_set = function( tbl ) if not tbl.name then return end tbl.cb = tbl.cb or spells[ tbl.name ] tbl.raw = tbl.raw or 0 tbl.ticks = tbl.ticks or 1 tbl.damage = tbl.damage or 0 tbl.healing = tbl.healing or 0 tbl.group_healing = tbl.group_healing or 0 tbl.cooldown = tbl.cooldown or 0 tbl.cd_remains = tbl.cooldown_remains or 0 tbl.start_cd = tbl.starts_cooldown or {} tbl.execute_time = max( 0, tbl.execute_time or 0 ) tbl.cost = tbl.cost or 0 tbl.secondary_cost = tbl.secondary_cost or 0 tbl.t_amp = tbl.t_amp or 1.0 tbl.delay = max( 0, tbl.delay or 0 ) -- Snapshot Channel Information if tbl.cb and tbl.raw > 0 and Player.channel.spellID and Player.channel.spellID == tbl.cb.spellID then Player.channel.action = tbl.cb Player.channel.raw = tbl.raw Player.channel.ticks = tbl.ticks end local raw_comp = tbl.raw / compression_value jeremy.raw[ tbl.name ] = raw_comp > 1 and floor( raw_comp ) or raw_comp if not tbl.background then local deficit = ( Player.primary_resource and Player.primary_resource.deficit ) or 0 local secondary_deficit = ( Player.secondary_resource and Player.secondary_resource.deficit ) or 0 local dpet = tbl.raw / max( 1, 1 + ( tbl.execute_time + tbl.delay ) * rate_time() ) tbl.d_time = dpet - tbl.cost tbl.d_cost = dpet / max( 1, 1 + deficit + tbl.cost ) / max( 1, 1 + secondary_deficit + tbl.secondary_cost ) local index = actionhash[ tbl.name ] or #actionlist+1 actionlist[ index ] = tbl actionhash[ tbl.name ] = index end aura_env.action_cache[ tbl.name ] = tbl end if spec == 0 then return end local targets_tod_range = 0 local targetList = {} local demiseList = {} local healer_targets = {} if fullUpdate then profilerStart() Player.Update() profilerEnd( "Update Unit Info" ) if Player.channel.action then action_set( { type = "channel", name = "channel_remaining", raw = Player.channel.raw or 0, execute_time = Player.channel.remaining + Player.channel.latency, cb = Player.channel.action, }) end local ParseAuras = function( Enemy, unitID ) -- Faux auras aura_env.targetAuras[ unitID ] = {} aura_env.targetAuras[ unitID ][ "priority_check" ] = { amp = Enemy.priority_modifier, expire = frameTime + 3600 } if Player.role == "TANK" then local threat_status = UnitThreatSituation( "player", unitID ) or 0 aura_env.targetAuras[ unitID ][ "threat_check" ] = { amp = ( threat_status < 3 and 2.0 or 1.0 ), expire = frameTime + 3600 } end -- Windwalker Specific if spec == aura_env.SPEC_INDEX[ "MONK_WINDWALKER" ] then -- Mark of the Crane Enemy.auraExists( 228287, function( auraData ) if auraData.sourceUnit == "player" then Player.motc_targets = Player.motc_targets + 1 return true end end ) -- Shadowflame Vulnerability Enemy.auraExists( 411376, function( auraData ) if auraData.sourceUnit == "player" then Player.sfv_targets = Player.sfv_targets + 1 return true end end ) -- Keefer's Skyreach Enemy.auraExists( 393047, function( auraData ) if auraData.sourceUnit == "player" then local expires = auraData.expirationTime if expires == 0 then expires = frameTime + 3600 end -- 0 amplifier for tracking purposes aura_env.targetAuras[ unitID ][ auraData.spellId ] = { amp = 0, expire = expires } return true end end ) -- Fae Exposure / Jadefire Brand Enemy.auraExists( { 356773, 395414 }, function( auraData ) if auraData.sourceUnit == "player" then local expires = auraData.expirationTime if expires == 0 then expires = frameTime + 3600 end Player.jfh_targets = Player.jfh_targets + 1 Player.jfh_dur_total = max( 0, Player.jfh_dur_total + ( expires - frameTime ) ) aura_env.targetAuras[ unitID ][ auraData.spellId ] = { amp = 1.12, expire = expires } return true end end ) end -- Brewmaster Specific if spec == aura_env.SPEC_INDEX[ "MONK_BREWMASTER" ] then -- Weapons of Order Debuff Enemy.auraExists( { 312106, 387179 }, function( auraData ) if auraData.sourceUnit == "player" then local expires = auraData.expirationTime if expires == 0 then expires = frameTime + 3600 end local woo_count = auraData.applications or 0 local woo_amp = 1 + ( 0.08 * woo_count ) aura_env.woo_best = max( aura_env.woo_best, woo_count ) Player.woo_targets = Player.woo_targets + 1 Player.woo_dur_total = max( 0, Player.woo_dur_total + ( expires - frameTime ) ) aura_env.targetAuras[ unitID ][ auraData.spellId ] = { amp = woo_amp, expire = expires } return true end end ) -- Keg Smash Enemy.auraExists( 121253, function( auraData ) if auraData.sourceUnit == "player" then Player.ks_targets = Player.ks_targets + 1 return true end end ) -- Breath of Fire Enemy.auraExists( 123725, function( auraData ) if auraData.sourceUnit == "player" then Player.bof_targets = Player.bof_targets + 1 return true end end ) end -- Generic -- Others defined in Init for id, aura_amp in pairs( aura_env.aura_amps ) do local copies = 0 Enemy.auraExists( id, function( auraData ) local expires = auraData.expirationTime if expires == 0 then expires = frameTime + 3600 end if copies > 0 then aura_env.targetAuras[ unitID ][ id ] = { amp = aura_env.targetAuras[ unitID ][ id ].amp * aura_amp.modifier, expire = min( aura_env.targetAuras[ unitID ][ id ].expires, expires ) } else aura_env.targetAuras[ unitID ][ id ] = { amp = aura_amp.modifier, expire = expires } end copies = copies + 1 if copies >= aura_amp.copies then return true end end ) end end aura_env.combatHealthRemaining = 0 aura_env.targetHealthRemaining = 0 aura_env.in_keystone = select( 10, GetInfo() ) == LE_SCENARIO_TYPE_CHALLENGE_MODE local UpdateTargets = function() local count = 0 local fight_remains = 0 local finalBoss = false aura_env.boss_lockdown = false aura_env.woo_best = 0 Player.motc_targets = 0 Player.jfh_targets = 0 Player.jfh_dur_total = 0 Player.ks_targets = 0 Player.bof_targets = 0 Player.sfv_targets = 0 Player.woo_targets = 0 aura_env.targetAuras = {} healer_targets = {} if spec == aura_env.SPEC_INDEX["MONK_MISTWEAVER"] then -- Skip this loop for efficiency if not MW Spec for it in WA_IterateGroupMembers() do if UnitIsDead( it ) == false and CheckInteractDistance( it, 1 ) then local deficit = UnitHealthMax( it ) - UnitHealth( it ) + UnitGetTotalHealAbsorbs( it ) if deficit > 0 then -- Valid healing target healer_targets[ #healer_targets + 1 ] = { deficit = deficit, unitID = it, } end end end end local targetLimit = 10 local configLimit = aura_env.config.target_limit if configLimit then if configLimit == 1 then targetLimit = 5 elseif configLimit == 2 then targetLimit = 10 elseif configLimit == 3 then targetLimit = 15 else targetLimit = 20 end end local TargetFilter = function() local validTargets = {} local checkTod = spells[ "touch_of_death" ] and Player.getCooldown( "touch_of_death" ) == 0 and aura_env.config.tod_glow > 1 for _, nameplateframe in pairs( GetNamePlates() ) do local unitID = nameplateframe.namePlateUnitToken local unitGUID = UnitGUID( unitID ) if unitGUID == UnitGUID( "target" ) then unitID = "target" end local glowTod = false if aura_env.validTarget( unitID ) then local enemy = aura_env.GetEnemy( unitGUID ) enemy.unitID = unitID enemy.range = aura_env.unitRange( unitID ) -- Update BigWig Intermission Timers local intermission = nil for k, v in pairs( aura_env.BW_intermission_timers ) do local remaining = v.expire - frameTime if v.encounterId ~= aura_env.encounter_id or remaining <= 0 then aura_env.BW_intermission_timers[ k ] = nil elseif UnitGUID( v.unitid ) == unitGUID then intermission = ( intermission and min( intermission, remaining ) ) or remaining end end enemy.intermission = intermission local excluded = ( enemy.priority_modifier == 0 ) if not excluded and enemy.lastHealth then excluded = enemy.lastHealth < 0.01 end if not excluded then excluded = enemy.range > 8 end if not excluded then if enemy.needsFullUpdate then ScanEvents( "UNIT_AURA_FULL_UPDATE", unitID ) end excluded = ( next( enemy.auraExclusions ) ~= nil ) end if excluded then aura_env.targetAuras[unitID] = {} aura_env.targetAuras[unitID]["priority_check"] = { amp = ( unitID == "target" and 0.01 ) or 0, expire = frameTime + 3600 } else -- ToD off CD and option enabled if checkTod then -- Valid ToD Target if ( enemy.healthPct < 0.15 and Player.talent.improved_touch_of_death.ok and aura_env.config.tod_glow < 3 ) or UnitHealth( unitID ) < UnitHealthMax( "player" ) then glowTod = true targets_tod_range = targets_tod_range + 1 end end validTargets[ #validTargets + 1 ] = enemy targetList[ #targetList + 1 ] = enemy.npcid end end if nameplateframe then if glowTod then LBG.PixelGlow_Start( nameplateframe, { 1, 0, 0, 1 } ) else LBG.PixelGlow_Stop( nameplateframe ) end end end if #validTargets > targetLimit then sort( validTargets, function( l, r ) if l.unitID == "target" then return true elseif r.unitID == "target" then return false elseif l.isBoss and not r.isBoss then return true elseif r.isBoss and not l.isBoss then return false elseif l.ttd and r.ttd then return l.ttd > r.ttd else return ( l.lastHealth or 1 ) > ( r.lastHealth or 1 ) end end ) end return validTargets end for _, enemy in pairs( TargetFilter() ) do if not aura_env.boss_lockdown then if enemy.isBoss then aura_env.boss_lockdown = true if not finalBoss and aura_env.in_keystone then local steps = select( 3, GetStepInfo() ) if steps then local _, _, _, _, total, _, _, current = GetCriteriaInfo( steps ) if current then current = tonumber( sub( current, 1, len( current ) - 1 ) ) or 0 if ( current / total ) > 0.99 then finalBoss = true end end end end end end count = count + 1 aura_env.combatHealthRemaining = aura_env.combatHealthRemaining + enemy.healthActual fight_remains = max( enemy.ttd or 0, fight_remains ) local demise = max( 1, floor( enemy.ttd or 1 ) ) demiseList[ demise ] = ( demiseList[ demise ] or 0 ) + 1 if enemy.unitID == "target" then aura_env.targetHealthRemaining = enemy.healthActual Player.react.spellsteal = next( enemy.stealable_auras ) ~= nil end ParseAuras( enemy, enemy.unitID ) if count >= targetLimit then break end end if finalBoss then aura_env.fight_remains = 3600 else if fight_remains > 0 then aura_env.fight_remains = max( 1, floor( fight_remains + 0.5 ) ) else aura_env.fight_remains = 300 end end sort( healer_targets, function( l, r ) return l.deficit > r.deficit end) return max( 1, count ) end if not aura_env.target_update or aura_env.target_update < frameTime - 1 then profilerStart() -- Nameplate iterator aura_env.target_count = UpdateTargets() aura_env.healer_targets = healer_targets profilerEnd( "UpdateTargets()" ) -- Force ST Config Option if aura_env.config.targetting == 2 then aura_env.target_count = 1 end profilerStart() -- Initialize one-time pull data if aura_env.target_count > 1 then aura_env.pull_hash = aura_env.hashEnemyList( targetList ) -- Generate unique pull ID aura_env.pull_data = aura_env.get_pull_data( aura_env.pull_hash ) -- Initialize pull data --aura_env.nextPullListener( aura_env.pull_hash ) end profilerEnd( "Hash Pull Data" ) profilerStart() -- Time based spawn / demise aura_env.raid_events = aura_env.EventTimers( demiseList ) profilerEnd( "Update Event Timers" ) aura_env.target_update = frameTime end end local t_spells = {} for name, action in pairs( spells ) do t_spells[ #t_spells + 1 ] = { name = name, action = action, } end local n_spells = #t_spells if n_spells > 0 and ( fullUpdate or not aura_env.actionlist_update or aura_env.actionlist_update < frameTime - ( 0.5 / n_spells ) )then aura_env.actionlist_update = frameTime debugprofilestart() local cpu_target = aura_env.config.cpu_target or 1.5 -- target CPU processing time in miliseconds if aura_env.rank_cpu_average then cpu_target = cpu_target - aura_env.rank_cpu_average end local process_trees = 5 if aura_env.action_cpu_average then process_trees = max( 1, floor( cpu_target / aura_env.action_cpu_average ) ) end if not InCombatLockdown() then -- limited to 5 out of combat process_trees = min( process_trees, 5 ) end local debug_process_min = nil local debug_process_max = nil if not aura_env.spell_range_min or aura_env.spell_range_min >= n_spells then aura_env.spell_range_min = 1 end aura_env.spell_range_max = ( aura_env.spell_range_min + process_trees ) for spell_it, _spell in ipairs( t_spells ) do local name = _spell.name local action = _spell.action local continue = false if spell_it < aura_env.spell_range_min or spell_it > aura_env.spell_range_max then if aura_env.action_cache[ name ] then action_set( aura_env.action_cache[ name ] ) continue = true end end if not continue then debug_process_min = debug_process_min or spell_it debug_process_max = spell_it local spellID = action.spellID local action_type = action.type if spellID ~= nil then -- Cache AP/SP values -- these sometimes change in PvP combat so we'll cache that as well if not action.pve_ap_value and not Player.is_pvp then action.pve_ap_value = ( type( action.ap ) == "function" and action.ap() or action.ap ) or 0 elseif not action.pvp_ap_value and Player.is_pvp then action.pvp_ap_value = ( type( action.ap ) == "function" and action.ap() or action.ap ) or 0 end if not action.pve_sp_value and not Player.is_pvp then action.pve_sp_value = ( type( action.sp ) == "function" and action.sp() or action.sp ) or 0 elseif not action.pvp_sp_value and Player.is_pvp then action.pvp_sp_value = ( type( action.sp ) == "function" and action.sp() or action.sp ) or 0 end local tooltip = 0 local ticks = ( type( action.ticks ) == "function" and action.ticks() or action.ticks ) or 1 local sp_mod = Player.is_pvp and action.pvp_sp_value or action.pve_sp_value if sp_mod > 0 then tooltip = ticks * ( sp_mod * aura_env.spell_power ) else local composite_attack_power = function( ap_type ) local base_power_mod = 1 local weapon_power = Player.weapon_power.main_hand if IsEquippedItemType( "Two-Hand" ) then if ap_type == "BOTH" then base_power_mod = 0.98 end elseif ap_type == "BOTH" then weapon_power = Player.weapon_power.both elseif ap_type == "OFFHAND" then weapon_power = Player.weapon_power.off_hand elseif ap_type == "NONE" then weapon_power = Player.weapon_power.none end return floor( Player.attack_power + weapon_power + 0.5 ) * base_power_mod end local ap_mod = Player.is_pvp and action.pvp_ap_value or action.pve_ap_value local attack_power = composite_attack_power( action.ap_type ) tooltip = ticks * ( attack_power * ap_mod ) end local action_targets = max( 1, min( 20, ( action.target_count and action.target_count() ) or 1 ) ) local action_delay = 0 local start_cooldown = action.start_cooldown or {} local combo_base = action.combo_base local true_name = gsub( combo_base or name, "_cancel", "" ) local action_cooldown = Player.getBaseCooldown( true_name ) local action_cd_remains = Player.getCooldown( true_name ) if combo_base and not start_cooldown[ combo_base ] then start_cooldown[ #start_cooldown + 1 ] = combo_base end local ready = Player.ready( action ) local execute_time = action.execute_time() local cost = nil local secondary_cost = nil if Player.primary_resource then if not Player.secondary_resource then secondary_cost = 0 end local costTable = GetSpellPowerCost( action.replaces or spellID ) if costTable then for _, costInfo in pairs( costTable ) do if cost and secondary_cost then break end if not cost and costInfo.type == Player.primary_resource.type then cost = costInfo.cost elseif not secondary_cost and costInfo.type == Player.secondary_resource.type then secondary_cost = costInfo.cost end end end cost = cost or 0 secondary_cost = secondary_cost or 0 end local gain = action.chi_gain and action.chi_gain() or 0 -- TODO: Need to rewrite this as generic resource gain somehow local secondary_gain = 0 action.base_gain = gain action.base_cost = cost action.base_secondary_cost = secondary_cost action.base_execute_time = execute_time if not ready then action_set( { type = action_type, name = name, raw = 0, cost = cost, execute_time = execute_time, } ) action.result = nil end if ready and action.skip_calcs then action_set( { type = action_type, name = name, raw = 1, cost = cost, execute_time = execute_time, } ) action.result = { damage = 1, healing = 1, } elseif ready then -- Action is not ready but is also not a background action if not action.background then --and not IsUsableSpell( spellID ) then local usable_in = 0 -- Add resource regeneration to action delay if low on resources if Player.primary_resource and Player.primary_resource.regen > 0 then if action.base_cost > 0 and Player.primary_resource.current < action.base_cost then local resource_delta = action.base_cost - Player.primary_resource.current usable_in = ( resource_delta / Player.primary_resource.regen ) end elseif Player.secondary_resource and Player.secondary_resource.regen > 0 then if secondary_cost > 0 and Player.secondary_resource.current < secondary_cost then local resource_delta = secondary_cost - Player.secondary_resource.current usable_in = ( resource_delta / Player.secondary_resource.regen ) end end -- Add cooldown time to action delay if cooldown is shorter than GCD local cd_after_gcd = action_cd_remains - Player.gcd_remains local charges = GetSpellCharges( spellID ) if cd_after_gcd > 0 and ( not charges or charges == 0 ) then usable_in = max( usable_in, cd_after_gcd ) end action_delay = max( action_delay, usable_in ) -- Check if we're channeling and add channel latency if Player.channel.remaining and action_delay > Player.channel.remaining then local latency = Player.channel.latency or 0 -- TODO: Generic usable during channel? Is this achievable via the API or spell labels? if Player.channel.spellID == 101546 and action.usable_during_sck then latency = 0 end action_delay = max( action_delay, latency ) end end -- Calculate total mitigation from action local actionMitigation = function( action, state ) local state = state or nil -- Stagger local stagger_reduction = 0 if action.reduce_stagger and type( action.reduce_stagger ) == "function" then stagger_reduction = action.reduce_stagger( state ) stagger_reduction = min( stagger_reduction, Player.stagger ) -- Limited to current stagger end -- Mitigation local mitigation = stagger_reduction if action.mitigate and type( action.mitigate ) == "function" then mitigation = mitigation + ( action.mitigate( state ) or 0 ) end return mitigation end -- -------------- local mitigation = actionMitigation( action, nil ) -- -------------- -- Action Multiplier tooltip = tooltip * Player.action_multiplier( action ) -- Cache spec auras -- these sometimes change in PvP so we will cache both if not action.aura_modifier_pve and not Player.is_pvp or not action.aura_modifier_pvp and Player.is_pvp then local total_aura_effect = aura_env.auraEffectForSpell( action.triggerSpell or spellID ) if Player.is_pvp then action.aura_modifier_pvp = total_aura_effect else action.aura_modifier_pve = total_aura_effect end end tooltip = tooltip * ( Player.is_pvp and action.aura_modifier_pvp or action.aura_modifier_pve ) -- Bonus damage and healing not related to ap or sp modifier local bonus_damage = action.bonus_da and action.bonus_da() or 0 local bonus_healing = action.bonus_heal and action.bonus_heal() or 0 -- Damage value local damage = action_type == "damage" and tooltip or 0 damage = damage + bonus_damage -- Self-healing value local healing = action_type == "self_heal" and tooltip or 0 healing = healing + bonus_healing -- Targeted healing value local group_healing = action_type == "smart_heal" and tooltip or 0 group_healing = group_healing + bonus_healing -- Versatility damage = damage * Player.vers_bonus healing = healing * Player.vers_bonus group_healing = group_healing * Player.vers_bonus -- Critical modifiers local crit_damage = 0 local crit_healing = 0 local crit_group_healing = 0 local crit_rate = 0 local crit_mod = 0 if action.may_crit then crit_rate = action.critical_rate and action.critical_rate() or Player.crit_bonus crit_mod = action.critical_modifier and action.critical_modifier() or 1 if Player.is_pvp then local pvp_crit_modifier = Player.spell.pvp_enabled.effectN( 3 ).mod crit_mod = crit_mod * pvp_crit_modifier end local crit_effect = crit_rate * crit_mod crit_damage = damage * crit_effect crit_healing = healing * crit_effect crit_group_healing = group_healing * crit_effect damage = damage + crit_damage healing = healing + crit_healing group_healing = group_healing + crit_group_healing end -- Target Effects local temporary_amplifiers = 1 local temporaryAmplifiers = function( action ) return global_modifier( action, action_delay ) * targetAuraEffect( action, action_delay ) end local target_multiplier = 1 local targetMultiplier = function( action ) local mul = 1 if action_type == "smart_heal" then mul = 0 for t_it = 1, action_targets do local target = aura_env.healer_targets[ t_it ] mul = mul + ( action.target_multiplier and action.target_multiplier( target ) or 1 ) end elseif action.target_multiplier then mul = action.target_multiplier( action_targets ) end return mul end -- Ability triggers ( GotD, Resonant Fists, etc., also used for combos ) local applyTriggerSpells = function( ) local damage_out, healing_out, group_heal_out, mitigate_out = 0, 0, 0, 0 local trigger_gain, trigger_cost, trigger_secondary_cost = 0, 0, 0 local trigger_time, trigger_delay = 0, 0 local trigger_cdr = {} local driver = action local driverName = combo_base or name driver.trigger = driver.trigger or {} driver.tick_trigger = driver.tick_trigger or {} local state_table = {} local trigger_spells = {} local trigger_exists = {} local trigger_pushback = function( spell, enabled, periodic, recursive_callback, stack ) local _driver = recursive_callback and spells[ recursive_callback ] or driver local _driverName = recursive_callback or driverName local _stack = ( stack or _driverName ) .. " -> " .. spell trigger_exists[ spell.."-".._driverName ] = true if _driver.result then -- Allow multiple identical triggers spell = gsub( spell, "%-.*", "" ) if spell and spells[ spell ] and spells[ spell ].result and spellRaw( spell ) > 0 then local _this = {} local callback_stack = {} local stack_driver = nil local triggerReady = function( self, state ) if type( enabled ) == "function" then return enabled( self, state ) end return enabled end local h = 5381 local l = nil for cb in gsub( _stack .. "->", "%s+", "" ):gmatch( "(.-)->" ) do l = ( stack_driver and h ) or nil for c in cb:gmatch( "." ) do h = ( bit.lshift( h, 5 ) + h ) + string.byte( c ) end insert( callback_stack, { name = cb, spell = spells[ cb ] or nil, driverName = stack_driver, result = spells[ cb ] and spells[ cb ].result or nil, } ) stack_driver = cb end _this.ready = triggerReady _this.stack = _stack _this.spell = spell _this.onTick = periodic _this.icd = spells[ spell ].icd or 0 _this.state = { -- Pass driver callbacks to trigger callback_state = state_table[ l ] or { time = execute_time + action_delay, primary = min( Player.primary_resource.max, ( Player.primary_resource.current + gain ) ) - cost, secondary = min( Player.secondary_resource.max, ( Player.secondary_resource.current + secondary_gain ) ) - secondary_cost, buffs = {}, cooldown = { [ true_name ] = max( 0, Player.getBaseCooldown( true_name ) - execute_time ) }, invalid = false, }, callback_stack = callback_stack, callback_name = _driverName, callback = _driver, result = _driver.result, ticks = ( _driver.result.ticks or 1 ) * ( _driver.result.target_count or 1 ), pos = #callback_stack, time = 0, primary = 0, secondary = 0, buffs = {}, cooldown = {}, invalid = false, } state_table[ h ] = _this.state _this.rate = spells[ spell ].trigger_rate or 1 if type( _this.rate ) == "function" then _this.rate = _this.rate( _this.state ) end if _this.rate > 0 then if not periodic then _this.state.count = _this.rate else if _this.icd > 0 then _this.state.count = min( _this.state.ticks, _this.rate * floor( _this.state.result.execute_time / _this.icd ) ) else _this.state.count = _this.rate * _this.state.ticks end end if Player.ready( spells[ _this.spell ], _this.state ) then trigger_spells[ #trigger_spells + 1 ] = _this end end end end end for spell, enabled in pairs( driver.trigger ) do trigger_pushback( spell, enabled, false ) end for spell, enabled in pairs( driver.tick_trigger ) do trigger_pushback( spell, enabled, true ) end -- Recursive triggers local do_recursion = true while ( do_recursion ) do do_recursion = false for _, trigger in pairs( trigger_spells ) do local spell = spells[ trigger.spell ] if spell.trigger then for recursive_trigger, enabled in pairs( spell.trigger ) do if not trigger_exists[ recursive_trigger.."-"..trigger.spell ] then trigger_pushback( recursive_trigger, enabled, false, trigger.spell, trigger.stack ) do_recursion = true end end end if spell.tick_trigger then for recursive_trigger, enabled in pairs( spell.tick_trigger ) do if not trigger_exists[ recursive_trigger.."-"..trigger.spell ] then trigger_pushback( recursive_trigger, enabled, true, trigger.spell, trigger.stack ) do_recursion = true end end end end end -- Setup state position and index tables sort( trigger_spells, function( l, r ) return l.state.pos < r.state.pos end) for _, trigger in pairs( trigger_spells ) do -- TODO: - Rename shadowed variables -- Initialization local state = trigger.state local driver = state.callback local driverName = state.callback_name local driverState = state.callback_state -- Inherit state from Driver if driverState then local copyState = function( e ) state[ e ] = driverState[ e ] end copyState( "primary" ) copyState( "secondary" ) copyState( "time" ) copyState( "buffs" ) copyState( "cooldown" ) end local result = state.result local spell = spells[ trigger.spell ] local spell_result = spell.result local tick_count = state.count local duration = result.execute_time local tick_damage = 0 local tick_healing = 0 local tick_group_heal = 0 local trigger_mitigate = 0 local trigger_damage = 0 local trigger_healing = 0 local trigger_group_heal = 0 local trigger_cdr = {} local basePrimary = state.primary local baseSecondary = state.secondary local baseTime = state.time if not state.invalid then -- Driver is a channeled ability and trigger is not background action if driver.channeled and not spell.background then local latency = Player.channel.latency or 0 -- Currently only relevant instance of this is spinning crane kick but use a generic action variable at some point local use_during_channel = driver.spellID == 101546 and spell.usable_during_sck if use_during_channel then latency = 0 local gcd = driver.gcd( driver, state ) local base_et = driver.execute_time( state ) state.time = state.time - ( base_et - gcd ) end trigger_delay = max( action_delay, latency ) - action_delay end state.time = max( state.time, trigger_delay + trigger_time ) local _, stack = ipairs( state.callback_stack ) if #stack > 1 then local _prev = #stack - 1 local _prev_cb = stack[ _prev ] if _prev_cb and _prev_cb.spell then local _prev_spell = _prev_cb.spell if ( not spell.background or not _prev_spell.background ) and _prev_spell.onExecute then _prev_spell.onExecute( _prev_spell, state ) end end end end -- Check ready state state.invalid = state.invalid or not trigger.ready( spell, state ) -- Check ready state state.invalid = state.invalid or not Player.ready( spell, state ) -- set init trigger CD local trigger_cd_remains = Player.getCooldown( trigger.spell, state ) local trigger_cd = Player.getBaseCooldown( trigger.spell ) if not state.invalid then -- Start Cooldown state.cooldown[ trigger.spell ] = trigger_cd local ticks = spell_result.ticks or 1 trigger_mitigate = actionMitigation( spell, state ) -- Tick Function local ticks_remaining = ticks local tick_time = spell.execute_time( state ) / ticks while ticks_remaining > 0 do local tick_partition = ticks_remaining < 2 and ticks_remaining or 1 -- Trigger is non-background action with execute time if tick_time > 0 and not spell.background then state.time = state.time + ( tick_time * tick_partition ) end tick_damage = ( spell_result.damage or 0 ) / ticks tick_healing = ( spell_result.healing or 0 ) / ticks tick_group_heal = ( spell_result.group_healing or 0 ) / ticks if tick_damage > 0 then local spell_am = Player.action_multiplier( spell ) local am_delta = Player.action_multiplier( spell, state ) if spell_am > 0 then am_delta = am_delta / spell_am end tick_damage = tick_damage * am_delta end trigger_damage = trigger_damage + ( tick_damage * tick_partition ) trigger_healing = trigger_healing + ( tick_healing * tick_partition ) trigger_group_heal = trigger_group_heal + ( tick_group_heal * tick_partition ) if spell.onImpact then for t_idx = 1, ( spell_result.target_count or 1 ) do spell.onImpact( spell, state ) end end ticks_remaining = ticks_remaining - tick_partition end -- Background spell multipliers if spell.background then local spell_target_multiplier = targetMultiplier( spell ) local spell_temporary_amplifiers = temporaryAmplifiers( spell ) trigger_damage = trigger_damage * spell_target_multiplier * spell_temporary_amplifiers trigger_healing = trigger_healing * spell_temporary_amplifiers trigger_group_heal = trigger_group_heal * spell_target_multiplier * spell_temporary_amplifiers end -- Trigger reduces cooldown of spell if spell.reduces_cd then for cdr_spell, cdr_value in pairs( spell.reduces_cd ) do cdr_spell = gsub( cdr_spell, "%-.*", "" ) local cdr = cdr_value if type( cdr ) == "function" then cdr = cdr( spell, state ) end if state.cooldown[ cdr_spell ] then state.cooldown[ cdr_spell ] = max( 0, state.cooldown[ cdr_spell ] - cdr ) end local remaining, total = Player.getCooldown( cdr_spell, state ) if cdr > 0 and remaining > 0 and total > 0 then state.cooldown[ trigger.spell ] = max( 0, state.cooldown[ trigger.spell ] - cdr ) local spell_action = aura_env.spells[ cdr_spell ] if spell_action and not spell_action.background then local mod_rate = aura_env.actionModRate( spell_action ) if mod_rate < 1 then cdr = cdr * ( 1 - mod_rate ) end local id = cdr_spell .. "-010" .. state.pos trigger_cdr[ id ] = ( trigger_cdr[ id ] or 0 ) + cdr end end end end damage_out = damage_out + trigger_damage * tick_count healing_out = healing_out + trigger_healing * tick_count group_heal_out = group_heal_out + trigger_group_heal * tick_count mitigate_out = mitigate_out + trigger_mitigate * tick_count end -- Update cooldown information for chained abilities if action.combo and not spell.background then action_cooldown = max( action_cooldown, trigger_cd ) action_cd_remains = max ( action_cd_remains, trigger_cd_remains ) trigger_delay = max( action_delay, action_cd_remains ) - action_delay start_cooldown[ #start_cooldown + 1] = trigger.spell end -- Update Resources state.primary = state.primary - ( spell.base_cost or 0 ) state.primary = state.primary + ( tick_count * ( spell.chi_gain and spell.chi_gain( state ) or 0 ) ) state.primary = min( state.primary, Player.primary_resource.max ) state.secondary = state.secondary - ( spell.base_secondary_cost or 0 ) --state.secondary = state.secondary + TODO Secondary Gain function state.secondary = min( state.secondary, Player.secondary_resource.max ) -- Update Trigger results trigger_cost = trigger_cost + ( basePrimary - state.primary ) trigger_secondary_cost = trigger_secondary_cost + ( baseSecondary - state.secondary ) trigger_time = trigger_time + ( state.time - baseTime ) end return { cost = trigger_cost, secondary_cost = trigger_secondary_cost, damage = damage_out, self_healing = healing_out, mitigation = mitigate_out, group_healing = group_heal_out, execute_time = trigger_time, delay = trigger_delay, cdr = trigger_cdr, } end local resultMatrix = {} local timeMatrix = { action_delay } local targetMatrix = { [ action_delay ] = action_targets } local triggerResults = {} local cacheResults = function( t ) resultMatrix[ t ] = { amplifier = temporary_amplifiers, callback = action, ticks = ticks, target_count = action_targets, damage = damage, self_healing = healing, group_healing = group_healing, mitigation = mitigation, crit_rate = crit_rate, crit_mod = crit_mod, critical_damage = crit_damage, critical_healing = crit_healing, critical_group_healing = crit_group_healing, execute_time = execute_time, delay = action_delay, trigger_results = triggerResults, cost = cost, secondary_cost = secondary_cost, reduce_cd = {}, } end if not action.background then -- Raid Events if action_type == "damage" and name ~= "touch_of_death" and action_cooldown > 0 and action.target_multiplier and ( not Player.channel.spellID or Player.channel.spellID ~= spellID ) then for _, raid_event in pairs( aura_env.raid_events ) do local count = raid_event.count local t = floor( raid_event.adds_in ) if count > action_targets and action_cooldown > t and t > action_delay then timeMatrix[ #timeMatrix + 1 ] = t targetMatrix[ t ] = count end end end local base_damage = damage local base_healing = healing local base_ghealing = group_healing for _, delay in pairs( timeMatrix ) do action_delay = delay action_targets = targetMatrix[ delay ] -- Target Multiplier target_multiplier = targetMultiplier( action ) -- Target Auras temporary_amplifiers = temporaryAmplifiers( action ) damage = base_damage * target_multiplier * temporary_amplifiers healing = base_healing * temporary_amplifiers group_healing = base_ghealing * target_multiplier * temporary_amplifiers -- Expel Harm -- TODO: Move to post processor if name == "expel_harm" then -- Only Healing config option for Brewmaster if spec ~= 1 or aura_env.config.eh_mode ~= 2 then local eh_damage = max( 1, 0.1 * healing ) damage = damage + eh_damage end end triggerResults = applyTriggerSpells( ) cacheResults( action_delay ) end else cacheResults( action_delay ) end local tmpD = nil local resultIdx = 0 local adjusted = 0 for idx, results in pairs( resultMatrix ) do results.result_base = { } for k, v in pairs( results ) do results.result_base[ k ] = v end -- Post-processing effects aura_env.actionPostProcessor( results ) -- Trigger effects for k, v in pairs( results.trigger_results ) do if results[ k ] then results[ k ] = results[ k ] + v end end if results.trigger_results.cdr then for s, r in pairs( results.trigger_results.cdr ) do results.reduce_cd[ s ] = r end end -- Healing Caps results.self_healing = min( results.self_healing, Player.health_deficit ) results.self_healing = max( results.self_healing, 0 ) local groupdeficit = 0 for t_it = 1, action_targets do groupdeficit = groupdeficit + ( aura_env.healer_targets[ t_it ] and aura_env.healer_targets[ t_it ].deficit or 0 ) end results.group_healing = min( results.group_healing, groupdeficit ) results.group_healing = max( results.group_healing, 0 ) -- Damage Caps if results.target_count == 1 and aura_env.targetHealthRemaining > 0 then results.damage = min ( aura_env.targetHealthRemaining, results.damage ) end if aura_env.combatHealthRemaining > 0 then results.damage = min ( aura_env.combatHealthRemaining, results.damage ) end -- ----------------------------------------------------- -- Adjust damage / heal value based on spec roles local D = results.damage if Player.role ~= "DAMAGER" or not IsInGroup() then D = ( D * 0.5 ) + results.self_healing if Player.role == "TANK" then D = D + results.mitigation else D = D + results.group_healing end end results.adjusted = D D = D / max( 1, 1 + results.execute_time + results.delay ) if not tmpD or tmpD < D then tmpD = D resultIdx = idx end end local result = resultMatrix[ resultIdx ] adjusted = result.adjusted or 0 cost = result.cost secondary_cost = result.secondary_cost ticks = result.ticks action_targets = result.target_count damage = result.damage healing = result.self_healing group_healing = result.group_healing mitigation = result.mitigation crit_rate = result.crit_rate crit_mod = result.crit_mod crit_damage = result.critical_damage crit_healing = result.critical_healing crit_group_healing = result.critical_group_healing execute_time = result.execute_time action_delay = result.delay temporary_amplifiers = result.amplifier action.result = result.result_base for s, v in pairs( result.reduce_cd ) do action.reduces_cd[ s ] = v end -- Cache curent player amplifier if name == default_action then Player.action_modifier = temporary_amplifiers end -- ----------------------------------------------------- -- Generic Brew CDR if spec == aura_env.SPEC_INDEX["MONK_BREWMASTER"] and action.brew_cdr then local brew_cdr = action.brew_cdr or 0 local brew_list = { "purifying_brew", "celestial_brew", "fortifying_brew", "black_ox_brew" } if type( brew_cdr ) == "function" then brew_cdr = brew_cdr( action ) end if brew_cdr > 0 then -- initialize table action.reduces_cd = action.reduces_cd or {} for _, brew in pairs ( brew_list ) do if spells[brew] and spells[brew].spellID then -- 000 is a unique identifier for general brew cdr -- this allows usage of both brew_cdr and reduces_cd for brew spells action.reduces_cd[brew.."-000"] = brew_cdr end end end end -- CDR if action.reduces_cd then local cdr_value = 0 local cdr_cost = 0 local cdr_time = 0 for spell, value in pairs( action.reduces_cd ) do local useResult = find( spell, "-010" ) spell = gsub( spell, "%-.*", "" ) local cdr = value local total_cdr = 0 if type(cdr) == "function" then cdr = cdr( action ) end local spell_action = spells[ spell ] local base_cd = Player.getBaseCooldown( spell ) if base_cd > 1 then if spell_action and cdr > 0 then if useResult then total_cdr = cdr else local timeLeft = Player.getCooldown( spell ) if timeLeft > 0 then total_cdr = min( cdr, timeLeft - execute_time ) local mod_rate = aura_env.actionModRate( spell_action ) if mod_rate < 1 then total_cdr = total_cdr * ( 1 - mod_rate ) end end end end if total_cdr > 0 then local spell_raw = spellRaw( spell ) local spell_time = spell_action.time_total or spell_action.execute_time() if spell_raw > adjusted then spell_raw = spell_raw - adjusted cdr_value = cdr_value + ( total_cdr / base_cd * spell_raw ) cdr_cost = cdr_cost + ( max( 0, ( spell_action.cost_total or 0 ) ) * total_cdr / base_cd ) cdr_time = cdr_time + ( spell_time * total_cdr / base_cd ) end end end end adjusted = adjusted + cdr_value cost = cost + cdr_cost execute_time = execute_time + cdr_time end -- Action has increased recharge rate local actionModRate = aura_env.actionModRate( action ) if action_cd_remains == 0 and action_cooldown > 0 and actionModRate < 1 then local duration = action_cooldown local recharge_cdr = min( duration, max( 0, action_cooldown - execute_time ) ) * ( 1 - actionModRate ) if recharge_cdr > 0 then local recast = 1 + ( recharge_cdr / action_cooldown ) adjusted = adjusted * recast cost = cost * recast execute_time = execute_time * recast end end -- Resource Gain if Player.primary_resource then if Player.primary_resource.regen > 0 then gain = gain + ( Player.primary_resource.regen * ( execute_time + action_delay ) ) end gain = min( Player.primary_resource.deficit, gain ) if aura_env.fight_remains > 5 then if Player.secondary_resource then if Player.secondary_resource.regen > 0 then secondary_gain = secondary_gain + ( Player.secondary_resource.regen * ( execute_time + action_delay ) ) end secondary_cost = secondary_cost - min( Player.primary_resource.deficit, secondary_gain ) end end end cost = cost - gain secondary_cost = secondary_cost - secondary_gain action.cost_total = cost action.time_total = execute_time action_set( { type = action_type, name = name, raw = adjusted, ticks = ticks, cost = cost, secondary_cost = secondary_cost, damage = damage, healing = healing, group_healing = group_healing, cooldown = action_cooldown, cooldown_remains = action_cd_remains, starts_cooldown = start_cooldown, delay = action_delay, base_cost = action.base_cost or 0, execute_time = execute_time, background = action.background, t_amp = temporary_amplifiers, combo_base = combo_base, }) end end end -- continue end aura_env.spell_range_min = aura_env.spell_range_min + process_trees local action_cpu_time = debugprofilestop() local actions_current = debug_process_max - debug_process_min local action_cpu_avg = action_cpu_time / actions_current if aura_env.action_cpu_average then aura_env.action_cpu_average = ( aura_env.action_cpu_average + action_cpu_avg ) / 2 else aura_env.action_cpu_average = action_cpu_avg end if action_cpu_time > 0 and profiler_enabled then aura_env.profiler_out[ "Update ActionList" ] = action_cpu_time end end if fullUpdate and next( actionlist ) ~= nil then -- Sort and rank candidate actions -- ----------------------------------------------------------- local t = min( 5, aura_env.fight_remains ) local a = Player.action_modifier local c = 0 local o = 0 local s = 0 local n = #actionlist local ow = {} if n > 0 then debugprofilestart() local scale_mode = 1 local hashed = {} local function sortActionList( list ) local maximum_sequence = 5 local minimum_step = 1 local sequence_n = 0 local sequence_t = {} local previous_s = s local non_op = false local pool = false o = global_modifier( default_action, t, false ) * targetAuraEffect( default_action, t ) if o > a then pool = true else if o < a then local bi_t = t / 2 local bi_o = global_modifier( default_action, bi_t, false ) * targetAuraEffect( default_action, bi_t ) if bi_o < a then t = bi_t end end end if not pool then -- Sort actions by time sort( list, function( l, r ) return l.d_time > r.d_time end) local validActions = {} for i = 1, n do local action = list[ i ] if not action or not action.cb or not action.cb.spellID or action.background or action.raw <= 0 or action.delay > t or not IsSpellKnown( action.cb.replaces or action.cb.spellID ) then list[ i ] = nil validActions[ i ] = nil else validActions[ i ] = action hashed[ i ] = list[ i ] end end if not next( list ) then return {} end while true do local delta_s = s - previous_s if non_op or s >= t or c <= 0 or ( delta_s > 0 and delta_s < minimum_step ) then break end previous_s = s local n_actions = #validActions if n_actions <= 1 then return list end non_op = true for j, action in pairs( validActions ) do local cd_remaining = ( ow[ action.name ] or action.cd_remains ) - s if action.cost <= c and cd_remaining <= 0 and action.delay <= ( t - s ) then non_op = false sequence_n = sequence_n + 1 sequence_t[ #sequence_t + 1 ] = action.name s = s + action.execute_time c = c - action.cost if s >= t or action.execute_time == 0 or action.cost <= 0 or sequence_n >= maximum_sequence then Player.action_sequence = sequence_t return list end if c <= 0 then break end ow[ action.name ] = action.cooldown for _, v in pairs( action.start_cd ) do local trigger = hashed[ actionhash[ v ] ] if trigger then ow[ trigger.name ] = trigger.cooldown end end elseif cd_remaining > t or action.delay > t then validActions[ j ] = nil end end end end -- Adjust by cost if Player.primary_resource then sort( list, function( l, r ) return l.d_cost > r.d_cost end ) local _, start = next( list ) Player.action_sequence = next( sequence_t ) ~= nil and sequence_t or { start.name } scale_mode = 0 end return list end actionlist = sortActionList( actionlist ) local debug_out = {} local action_debug = false for k, v in pairs( actionlist ) do if not v.raw or v.raw == 0 then jeremy.rank[ v.name ] = 0 elseif v.base_cost and Player.primary_resource and v.base_cost > Player.primary_resource.current then jeremy.rank[ v.name ] = 0 else local action_name = gsub( v.combo_base and v.combo_base or v.name, "_cancel", "" ) jeremy.rank[ action_name ] = jeremy.rank[ action_name ] or k jeremy.scale[ action_name ] = jeremy.scale[ action_name ] or ( scale_mode == 1 and v.d_time or v.d_cost ) if action_debug and k <= 5 then debug_out[ k ] = { name = v.name, dpet = v.d_time, raw = v.raw, cost = v.cost, s_cost = v.secondary_cost, execute_time = v.execute_time, delay = v.delay, } end end end if next( debug_out ) ~= nil then ScanEvents( "JEREMY_DEBUG_PRIORITY", debug_out ) end local rank_cpu_time = debugprofilestop() local actions_current = n local rank_cpu_avg = rank_cpu_time / actions_current if aura_env.rank_cpu_average then aura_env.rank_cpu_average = ( aura_env.rank_cpu_average + rank_cpu_avg ) / 2 else aura_env.rank_cpu_average = rank_cpu_avg end if rank_cpu_time > 0 and profiler_enabled then aura_env.profiler_out[ "Rank Candidate Actions" ] = rank_cpu_time end end -- ----------------------------------------------------------- -- Ability options jeremy.boss_lockdown = aura_env.boss_lockdown jeremy.fight_remains = aura_env.fight_remains jeremy.target_ttd = aura_env.target_ttd if aura_env.target_count > 1 and targets_tod_range == 1 and aura_env.target_ttd <= 1 then jeremy.force_tod = true else jeremy.force_tod = false end local woo_buff = ( spec == aura_env.SPEC_INDEX[ "MONK_BREWMASTER" ] and Player.findAura( 387184 ) ) or nil if woo_buff and woo_buff.remaining <= ( 0.25 + Player.gcd_duration ) * ( 4 - aura_env.woo_best ) then jeremy.woo_prio = true else jeremy.woo_prio = false end -- end of ability options -- ------------------------------------------------------------- -- Passed Configuration Options jeremy.options = {} jeremy.is_beta = Player.is_beta() jeremy.options.limit = aura_env.config.limit or 5 jeremy.options.scaling = aura_env.config.scaling_option or 1 jeremy.options.inverse = ( aura_env.config.inverse == 2 ) or false jeremy.options.hold_sef = aura_env.config.hold_sef or 1 aura_env.jeremy_update = jeremy ScanEvents( "JEREMY_UPDATE", jeremy ) return true end return false end if event == "ENCOUNTER_START" then aura_env.encounter_id = tostring( ... ) return false elseif event == "ENCOUNTER_END" then aura_env.encounter_id = nil return false end if event == "JEREMY_STARTBAR" or event == "RELOE_SPELLCD_STATE_UPDATE" then local key, time, spell, srcGUID if event == "RELOE_SPELLCD_STATE_UPDATE" then spell, srcGUID, key, time = ... if not ( spell and time ) then return end else key = select( 3, ... ) time = select( 5, ... ) end key = tostring( key ) if key and time then local config = aura_env.bw_config[ key ] if config and config.enabled then local expirationTime = frameTime + time local config_type = config.type or "ERROR" local t_key = (srcGUID or "SB").."-"..key if config_type == "ADD_SPAWN" then aura_env.BW_add_timers[ t_key ] = { key = key, expire = expirationTime, count = config.count or 0, encounterId = aura_env.encounter_id or -1, srcGUID = srcGUID, type = config_type, } elseif config_type == "INTERMISSION" then aura_env.BW_intermission_timers[ t_key ] = { key = key, expire = expirationTime, unitid = config.unitid or "boss1", encounterId = aura_env.encounter_id or -1, srcGUID = srcGUID, type = config_type, } elseif config_type == "TANKBUSTER" then local affects = config.affects or 1 if affects == 1 and Player.role == "TANK" or affects == 2 and Player.role ~= "TANK" or affects == 3 then -- Send message to Raid Ability Timeline if event == "RELOE_SPELLCD_STATE_UPDATE" then local Enemy = aura_env.GetEnemy( srcGUID ) ScanEvents( "JEREMY_TIMELINE_UPDATE", spell, srcGUID, key, time, Enemy.marker ) end aura_env.BW_buster_timers [ t_key ] = { key = key, expire = expirationTime, damage_type = config.damage_type or 1, encounterId = aura_env.encounter_id or -1, srcGUID = srcGUID, type = config_type, } end end end end return false end if event == "UNIT_SPELLCAST_SUCCEEDED" then local _, _, spellID = ... if spellID == 137639 then aura_env.sef_fixate = nil elseif spellID == 221771 then local dstGUID = UnitGUID( "target" ) aura_env.sef_fixate = dstGUID aura_env.last_fixate_bonus = aura_env.forwardModifier( aura_env.spells["tiger_palm"], 1 ) end if spellID and aura_env.combo_strike[spellID] then if Player.last_combo_strike ~= spellID then Player.last_combo_strike = spellID aura_env.fast = true end end return false end if event == "COMBAT_RATING_UPDATE" or event == "PLAYER_ENTERING_WORLD" then aura_env.initGear() end if event == "TRAIT_CONFIG_UPDATED" or event == "PLAYER_SPECIALIZATION_CHANGED" or event == "PLAYER_ENTERING_WORLD" then aura_env.initSpecialization() end if event == "PLAYER_ENTERING_WORLD" or event == "PLAYER_REGEN_ENABLED" then aura_env.updateDB() Combat = { avg_level = 0, damage_by_level = 0, damage_taken = 0, damage_taken_avoidable = 0, damage_taken_unavoidable = 0, recent_damage = {}, } aura_env.CURRENT_MARKERS = {} end -- Everything below returns false -- ------------------------------ if event == "UNIT_HEALTH" or event == "UNIT_ABSORB_AMOUNT_CHANGED" then local unitID = ... local unitGUID = UnitGUID( unitID ) if aura_env.validTarget( unitID ) then local Enemy = aura_env.GetEnemy( unitGUID ) Enemy.healthActual = UnitHealth( unitID ) - ( ( aura_env.earlyDeath[ Enemy.npcid ] or 0 ) * UnitHealthMax( unitID ) ) + UnitGetTotalAbsorbs( unitID ) Enemy.healthPct = min( 1, Enemy.healthActual / UnitHealthMax( unitID ) ) if Enemy.lastHealth and Enemy.lastSeen then local difference = Enemy.lastHealth - Enemy.healthPct if difference > 0 then local elapsed = frameTime - Enemy.lastSeen if not Enemy.n or Enemy.n == 0 then Enemy.rate = difference / elapsed Enemy.n = 1 else local samples = min( Enemy.n, 9 ) local newRate = Enemy.rate * samples + ( difference / elapsed ) Enemy.n = samples + 1 Enemy.rate = newRate / Enemy.n end end end Enemy.lastHealth = Enemy.healthPct Enemy.lastSeen = frameTime if Enemy.rate then Enemy.ttd = Enemy.healthPct / Enemy.rate if Enemy.intermission then Enemy.ttd = min( Enemy.ttd, Enemy.intermission ) end if unitGUID == UnitGUID( "target" ) then aura_env.target_ttd = max( 1, floor( Enemy.ttd + 0.5 ) ) aura_env.taret_abs = UnitGetTotalAbsorbs( unitID ) or 0 end end end return false end if event == "NAME_PLATE_UNIT_ADDED" or event == "NAME_PLATE_UNIT_UPDATED" or event == "UPDATE_MOUSEOVER_UNIT" then local unitID, unitGUID if event == "UPDATE_MOUSEOVER_UNIT" then -- Only use this event if there is no nameplate available -- and there is a valid mouseover unitID = "mouseover" unitGUID = UnitGUID( "mouseover" ) if GetNamePlateForUnit( "mouseover" ) then return end else unitID = ... unitGUID = ( unitID and UnitGUID( unitID ) ) or nil end if not unitGUID then return end if UnitExists( unitID ) then local Enemy = aura_env.GetEnemy( unitGUID ) if not Enemy.isBoss then if unitGUID == UnitGUID( "boss1" ) or unitGUID == UnitGUID( "boss2" ) or unitGUID == UnitGUID( "boss3" ) or unitGUID == UnitGUID( "boss4" ) or unitGUID == UnitGUID( "boss5" ) then Enemy.isBoss = true end end local oldMarker = Enemy.marker Enemy.marker = GetRaidTargetIndex( unitID ) Enemy.level = UnitLevel( unitID ) Enemy.priority_modifier = aura_env.npc_priority[ Enemy.npcid ] or 1.0 local am_level = aura_env.config.automarker_enable local am_enable = am_level == 2 or ( am_level == 3 and Player.role == "TANK" ) or ( am_level == 4 and Player.leader ) if aura_env.AUTOMARKER and am_enable then local value = aura_env.AUTOMARKER[ UnitName( unitID ) ] or aura_env.AUTOMARKER[ Enemy.npcid ] if value then local mark = find( value, "MARK" ) ~= nil Enemy.interruptTarget = find( value, "INTERRUPT" ) ~= nil Enemy.stunTarget = find( value, "STUN" ) ~= nil local l, h = 1, 6 if find( value, "KILL" ) then l, h = 7, 8 end if Enemy.marker then aura_env.CURRENT_MARKERS[ Enemy.marker ] = unitGUID elseif mark == true then local marks_available = false for m = h, l, -1 do if aura_env.CURRENT_MARKERS[ m ] == nil then marks_available = true break end end for m = h, l, -1 do -- We need a mark, can we steal one? if not marks_available and aura_env.CURRENT_MARKERS[ m ] then local marked_enemy = aura_env.GetEnemy( aura_env.CURRENT_MARKERS[ m ] ) -- Do not steal from enemy in combat if not marked_enemy.inCombat then if Enemy.inCombat or Enemy.range < marked_enemy.range then aura_env.CURRENT_MARKERS[ m ] = nil end end end if aura_env.CURRENT_MARKERS[ m ] == nil then SetRaidTarget( unitID, m ) aura_env.CURRENT_MARKERS[ m ] = unitGUID Enemy.marker = m break end end end end end if Enemy.marker ~= oldMarker then ScanEvents( "JEREMY_MARKER_CHANGED", unitGUID, Enemy.marker ) end end return false end if event == "UNIT_AURA" or event == "UNIT_AURA_FULL_UPDATE" then local unitID, updateInfo = ... local unitGUID = UnitGUID( unitID ) if unitID == "player" or aura_env.validTarget( unitID ) then local Unit = ( unitID == "player" and Player ) or aura_env.GetEnemy( unitGUID ) local parseAuraData = function( auraData ) auraData.name = gsub( lower( auraData.name ), "%s+", "_" ) auraData.stacks = auraData.applications or 0 auraData.duration = auraData.duration or 0 return auraData end if event == "UNIT_AURA_FULL_UPDATE" then if not Unit.lastFullUpdate or Unit.lastFullUpdate < frameTime - aura_env.update_rate then Unit.auraExclusions = {} Unit.auraDataByInstance = {} Unit.auraInstancesByID = {} -- Monk Diffuse Magic if Unit == Player then Unit.diffuse_auras = {} Unit.diffuse_reflects = {} else Unit.stealable_auras = {} end local auraSlots = { UnitAuraSlots( unitID, "HELPFUL|HARMFUL" ) } for slot in pairs( auraSlots ) do local auraData = GetAuraDataBySlot( unitID, slot ) if auraData then local instanceID = auraData.auraInstanceID local spellId = auraData.spellId if aura_env.auraExclusions[ spellId ] then Unit.auraExclusions[ instanceID ] = spellId end -- Monk Diffuse Magic if Unit == Player then local valid_diffuse = aura_env.diffuse_options[ spellId ] if valid_diffuse and valid_diffuse.enabled then Unit.diffuse_auras[ instanceID ] = true if valid_diffuse.reflect then Unit.diffuse_reflects[ instanceID ] = true end end else if auraData.isStealable then Unit.stealable_auras[ instanceID ] = true end end Unit.auraInstancesByID[ spellId ] = Unit.auraInstancesByID[ spellId ] or {} Unit.auraInstancesByID[ spellId ][ instanceID ] = true Unit.auraDataByInstance[ instanceID ] = parseAuraData( auraData ) end end Unit.needsFullUpdate = false Unit.lastFullUpdate = frameTime end else if updateInfo.isFullUpdate then Unit.needsFullUpdate = true else if updateInfo.addedAuras then for _, auraData in pairs( updateInfo.addedAuras ) do local instanceID = auraData.auraInstanceID local spellId = auraData.spellId if aura_env.auraExclusions[ spellId ] then Unit.auraExclusions[ instanceID ] = spellId end -- Monk Diffuse Magic if Unit == Player then local valid_diffuse = aura_env.diffuse_options[ spellId ] if valid_diffuse and valid_diffuse.enabled then Unit.diffuse_auras[ instanceID ] = true if valid_diffuse.reflect then Unit.diffuse_reflects[ instanceID ] = true end end else if auraData.isStealable then Unit.stealable_auras[ instanceID ] = true end end Unit.auraInstancesByID[ spellId ] = Unit.auraInstancesByID[ spellId ] or {} Unit.auraInstancesByID[ spellId ][ instanceID ] = true Unit.auraDataByInstance[ instanceID ] = parseAuraData( auraData ) end end if updateInfo.updatedAuraInstanceIDs then for _, instanceID in pairs( updateInfo.updatedAuraInstanceIDs ) do local auraData = GetAuraDataByAuraInstanceID( unitID, instanceID ) if auraData then auraData.name = gsub( lower( auraData.name ), "%s+", "_" ) auraData.stacks = auraData.applications or 0 local spellId = auraData.spellId Unit.auraInstancesByID[ spellId ] = Unit.auraInstancesByID[ spellId ] or {} Unit.auraInstancesByID[ spellId ][ instanceID ] = true Unit.auraDataByInstance[ instanceID ] = parseAuraData( auraData ) end end end if updateInfo.removedAuraInstanceIDs then for _, instanceID in pairs( updateInfo.removedAuraInstanceIDs ) do Unit.auraExclusions[ instanceID ] = nil -- Monk Diffuse Magic if Unit == Player then Unit.diffuse_auras[ instanceID ] = nil Unit.diffuse_reflects[ instanceID ] = nil else Unit.stealable_auras[ instanceID ] = nil end if Unit.auraDataByInstance[ instanceID ] then local spellId = Unit.auraDataByInstance[ instanceID ].spellId Unit.auraInstancesByID[ spellId ][ instanceID ] = nil Unit.auraDataByInstance[ instanceID ] = nil end end end end end end return false end if event == "UNIT_THREAT_LIST_UPDATE" or event == "NAME_PLATE_UNIT_REMOVED" then local unitID = ... local unitGUID = UnitGUID( unitID ) if not unitGUID then return end if not UnitExists( unitID ) then aura_env.ResetEnemy( unitGUID ) else local Enemy = aura_env.GetEnemy( unitGUID ) if UnitAffectingCombat( unitID ) then Enemy.inCombat = true elseif Enemy.inCombat then local unitInCombat = false for it in WA_IterateGroupMembers() do if UnitThreatSituation( it, unitID ) then unitInCombat = true break end end if not unitInCombat then aura_env.ResetEnemy( unitGUID ) end end if Enemy.inCombat and not Enemy.combatStart then Enemy.combatStart = frameTime end end ScanEvents( "NAME_PLATE_UNIT_UPDATED", unitID ) return false end if event == "COMBAT_LOG_EVENT_UNFILTERED" then local LogDamage = function (_, eventtype, _, srcGUID, srcName, srcFlags, _, dstGUID, ...) if eventtype == nil or eventtype == "ENVIRONMENTAL_DAMAGE" or not find( eventtype, "_DAMAGE" ) then return false end local enemyGUID = nil if dstGUID == UnitGUID( "player" ) then enemyGUID = srcGUID else local mask = COMBATLOG_OBJECT_AFFILIATION_MASK local bitfield = bit_band( srcFlags, mask ) if srcGUID == UnitGUID( "player" ) or srcGUID == UnitGUID( "pet" ) or bitfield == COMBATLOG_OBJECT_AFFILIATION_MINE then enemyGUID = dstGUID end end if enemyGUID == nil then return false end local Enemy = aura_env.GetEnemy( enemyGUID ) local spellID, spellSchool, amount if eventtype == "SWING_DAMAGE" then _, _, _, amount = ... else _, _, _, spellID, _, spellSchool, amount = ... end spellSchool = spellSchool or 1 spellID = spellID or 0 if amount == nil or type( amount ) ~= "number" or amount == 0 then return false end if srcGUID == enemyGUID then if dstGUID == UnitGUID( "player" ) and Enemy.level > 0 then Combat.damage_by_level = Combat.damage_by_level + ( Enemy.level * amount ) Combat.damage_taken = Combat.damage_taken + amount Combat.avg_level = Combat.damage_by_level / Combat.damage_taken; Combat.recent_damage[ #Combat.recent_damage + 1 ] = { amount = amount, expire = frameTime + aura_env.RECENT_DURATION } if spellSchool == 1 and not find( eventtype, "PERIODIC" ) then Combat.damage_taken_avoidable = Combat.damage_taken_avoidable + amount else Combat.damage_taken_unavoidable = Combat.damage_taken_unavoidable + amount end end else -- Not pet damage if srcGUID == UnitGUID( "player" ) then -- spell tracking ... if spellID and aura_env.pull_hash ~= "" then aura_env.coneTickListener( spellID, enemyGUID ) end end end end local LogDeath = function (_, eventtype, _, srcGUID, _, _, _, dstGUID, ...) if eventtype == "UNIT_DIED" or eventtype == "UNIT_DESTROYED" or eventtype == "UNIT_DISSIPATES" then aura_env.ResetEnemy( dstGUID, true ) end end if InCombatLockdown() then LogDamage( CombatLogGetCurrentEventInfo() ) LogDeath( CombatLogGetCurrentEventInfo() ) end return false end return false end
Leave a Comment