Possibly massive improvements to performance in raid, start on notification system

notifications
4n0n 2 years ago
parent de09096a71
commit 1ba0cd7115
  1. 68
      scripts/restodruid.lua
  2. 187
      src/AuraTable/AuraTable.lua
  3. 124
      src/NotificationsList/NotificationsList.lua
  4. 43
      src/Spell/Spell.lua
  5. 48
      src/Unit/Unit.lua
  6. 24
      src/UnitManager/UnitManager.lua
  7. 8
      src/_bastion.lua

@ -139,7 +139,7 @@ local Lowest = Bastion.UnitManager:CreateCustomUnit('lowest', function(unit)
return lowest return lowest
end) end)
local DispelTarget = Bastion.UnitManager:CreateCustomUnit('rejuv', function(unit) local DispelTarget = Bastion.UnitManager:CreateCustomUnit('dispel', function(unit)
local lowest = nil local lowest = nil
local lowestHP = math.huge local lowestHP = math.huge
@ -366,7 +366,7 @@ local PLACE_EFFLO = false
RestoCommands:Register('efflo', 'Request the engine to place an Efflorescence', function() RestoCommands:Register('efflo', 'Request the engine to place an Efflorescence', function()
PLACE_EFFLO = true PLACE_EFFLO = true
Bastion:Print('Efflorescence will be placed on next cast') Bastion.Notifications:AddNotification(Efflorescence:GetIcon(), "Efflorescence requested")
end) end)
local DefaultAPL = Bastion.APL:New('default') local DefaultAPL = Bastion.APL:New('default')
@ -515,18 +515,6 @@ DefaultAPL:AddSpell(
end):SetTarget(RejuvUnit) end):SetTarget(RejuvUnit)
) )
DefaultAPL:AddSpell(
Regrowth:CastableIf(function(self)
return Lowest:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling()
and Player:CanSee(Lowest) and
(
not Player:GetAuras():FindMy(Regrowth):IsUp() and Lowest:GetHP() < 70 or
(Lowest:GetHP() <= 85 and Player:GetAuras():FindMy(ClearCasting):IsUp())) and
not Player:GetAuras():FindMy(SoulOfTheForest):IsUp() and
not Player:IsMoving()
end):SetTarget(Lowest)
)
DefaultAPL:AddSpell( DefaultAPL:AddSpell(
Lifebloom:CastableIf(function(self) Lifebloom:CastableIf(function(self)
return Player:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling() return Player:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling()
@ -547,42 +535,54 @@ DefaultAPL:AddSpell(
end):SetTarget(Tank) end):SetTarget(Tank)
) )
DefaultAPL:AddSpell(
Regrowth:CastableIf(function(self)
return Lowest:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling()
and Player:CanSee(Lowest) and
(
not Player:GetAuras():FindMy(Regrowth):IsUp() and Lowest:GetHP() < 70 or
(Lowest:GetHP() <= 85 and Player:GetAuras():FindMy(ClearCasting):IsUp())) and
not Player:GetAuras():FindMy(SoulOfTheForest):IsUp() and
not Player:IsMoving()
end):SetTarget(Lowest)
)
DefaultAPL:AddSpell( DefaultAPL:AddSpell(
Sunfire:CastableIf(function(self) Sunfire:CastableIf(function(self)
return Target:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling() return Bastion.UnitManager['target']:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling()
and Player:CanSee(Target) and and Player:CanSee(Bastion.UnitManager['target']) and
( (
not Target:GetAuras():FindMy(SunfireAura):IsUp() or not Bastion.UnitManager['target']:GetAuras():FindMy(SunfireAura):IsUp() or
Target:GetAuras():FindMy(SunfireAura):GetRemainingTime() <= 5.4) and Target:IsHostile() and Bastion.UnitManager['target']:GetAuras():FindMy(SunfireAura):GetRemainingTime() <= 5.4) and Bastion.UnitManager['target']:IsHostile() and
Target:IsAffectingCombat() Bastion.UnitManager['target']:IsAffectingCombat()
end):SetTarget(Target) end):SetTarget(Bastion.UnitManager['target'])
) )
DefaultAPL:AddSpell( DefaultAPL:AddSpell(
Moonfire:CastableIf(function(self) Moonfire:CastableIf(function(self)
return Target:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling() return Bastion.UnitManager['target']:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling()
and Player:CanSee(Target) and and Player:CanSee(Bastion.UnitManager['target']) and
( (
not Target:GetAuras():FindMy(MoonfireAura):IsUp() or not Bastion.UnitManager['target']:GetAuras():FindMy(MoonfireAura):IsUp() or
Target:GetAuras():FindMy(MoonfireAura):GetRemainingTime() <= 5.4) and Target:IsHostile() and Bastion.UnitManager['target']:GetAuras():FindMy(MoonfireAura):GetRemainingTime() <= 5.4) and Bastion.UnitManager['target']:IsHostile() and
Target:IsAffectingCombat() Bastion.UnitManager['target']:IsAffectingCombat()
end):SetTarget(Target) end):SetTarget(Bastion.UnitManager['target'])
) )
DefaultAPL:AddSpell( DefaultAPL:AddSpell(
Starsurge:CastableIf(function(self) Starsurge:CastableIf(function(self)
return Target:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling() return Bastion.UnitManager['target']:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling()
and Player:CanSee(Target) and Target:IsHostile() and and Player:CanSee(Bastion.UnitManager['target']) and Bastion.UnitManager['target']:IsHostile() and
Target:IsAffectingCombat() Bastion.UnitManager['target']:IsAffectingCombat()
end):SetTarget(Target) end):SetTarget(Bastion.UnitManager['target'])
) )
DefaultAPL:AddSpell( DefaultAPL:AddSpell(
Wrath:CastableIf(function(self) Wrath:CastableIf(function(self)
return Target:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling() return Bastion.UnitManager['target']:Exists() and self:IsKnownAndUsable() and not Player:IsCastingOrChanneling()
and Player:CanSee(Target) and not Player:IsMoving() and Target:IsHostile() and and Player:CanSee(Bastion.UnitManager['target']) and not Player:IsMoving() and Bastion.UnitManager['target']:IsHostile() and
Target:IsAffectingCombat() Bastion.UnitManager['target']:IsAffectingCombat()
end):SetTarget(Target) end):SetTarget(Bastion.UnitManager['target'])
) )
RestoModule:Sync(function() RestoModule:Sync(function()

@ -17,47 +17,44 @@ function AuraTable:New(unit)
local self = setmetatable({}, AuraTable) local self = setmetatable({}, AuraTable)
self.unit = unit self.unit = unit
self.buffs = {}
self.debuffs = {}
self.auras = {} self.auras = {}
-- Our player is usually the most important unit, so we cache the auras for it
self.playerAppliedBuffs = {}
self.playerAppliedDebuffs = {}
self.playerAuras = {} self.playerAuras = {}
self.guid = unit:GetGUID() self.guid = unit:GetGUID()
self.instanceIDLookup = {} self.instanceIDLookup = {}
Bastion.EventManager:RegisterWoWEvent('UNIT_AURA', function(unit, auras) return self
local u = Bastion.UnitManager[unit]
if not self.unit:IsUnit(u) then
return
end end
function AuraTable:OnUpdate(auras)
local isFullUpdate = auras.isFullUpdate local isFullUpdate = auras.isFullUpdate
if isFullUpdate then if isFullUpdate then
self:Update() self:Update()
print("Full update requested for " .. unit)
return return
end end
local addedAuras = auras.addedAuras
local removedAuras = auras.removedAuraInstanceIDs local removedAuras = auras.removedAuraInstanceIDs
local addedAuras = auras.addedAuras
local updatedAuras = auras.updatedAuraInstanceIDs local updatedAuras = auras.updatedAuraInstanceIDs
-- Add auras
if addedAuras and #addedAuras > 0 then
for i = 1, #addedAuras do
local aura = Bastion.Aura:CreateFromUnitAuraInfo(addedAuras[i])
self:AddOrUpdateAuraInstanceID(aura:GetAuraInstanceID(), aura)
end
end
-- DevTools_Dump(addedAuras) -- DevTools_Dump(addedAuras)
if updatedAuras and #updatedAuras > 0 then if updatedAuras and #updatedAuras > 0 then
for i = 1, #updatedAuras do for i = 1, #updatedAuras do
local id = updatedAuras[i] local id = updatedAuras[i]
local newAura = C_UnitAuras.GetAuraDataByAuraInstanceID(unit, id); local newAura = C_UnitAuras_GetAuraDataByAuraInstanceID(self.unit.unit, id);
if newAura then if newAura then
local aura = Bastion.Aura:CreateFromUnitAuraInfo(newAura) local aura = Bastion.Aura:CreateFromUnitAuraInfo(newAura)
self:UpdateInstanceID(id, aura) self:AddOrUpdateAuraInstanceID(aura:GetAuraInstanceID(), aura)
else
print("Instance ID " .. id .. " not found" .. " for unit " .. unit)
end end
end end
end end
@ -68,49 +65,6 @@ function AuraTable:New(unit)
self:RemoveInstanceID(removedAuras[i]) self:RemoveInstanceID(removedAuras[i])
end end
end end
-- Add auras
if addedAuras and #addedAuras > 0 then
for i = 1, #addedAuras do
local aura = Bastion.Aura:CreateFromUnitAuraInfo(addedAuras[i])
if aura:IsBuff() then
if aura:GetSource():Exists() and aura:GetSource():IsUnit(Bastion.UnitManager['player']) then
if not self.playerAppliedBuffs[aura:GetSpell():GetID()] then
self.playerAppliedBuffs[aura:GetSpell():GetID()] = {}
end
self.playerAppliedBuffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'playerAppliedBuffs',
aura:GetSpell():GetID() }
else
if not self.buffs[aura:GetSpell():GetID()] then
self.buffs[aura:GetSpell():GetID()] = {}
end
self.buffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'buffs',
aura:GetSpell():GetID() }
end
else
if aura:GetSource():Exists() and aura:GetSource():IsUnit(Bastion.UnitManager['player']) then
if not self.playerAppliedDebuffs[aura:GetSpell():GetID()] then
self.playerAppliedDebuffs[aura:GetSpell():GetID()] = {}
end
self.playerAppliedDebuffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'playerAppliedDebuffs',
aura:GetSpell():GetID() }
else
if not self.debuffs[aura:GetSpell():GetID()] then
self.debuffs[aura:GetSpell():GetID()] = {}
end
self.debuffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'debuffs', aura:GetSpell():GetID() }
end
end
end
end
end)
return self
end end
function AuraTable:RemoveInstanceID(instanceID) function AuraTable:RemoveInstanceID(instanceID)
@ -118,106 +72,74 @@ function AuraTable:RemoveInstanceID(instanceID)
return return
end end
local t, id = unpack(self.instanceIDLookup[instanceID]) local id = self.instanceIDLookup[instanceID]
-- print("Removing aura from table: " .. t .. " " .. id .. " " .. index .. "")
local a = self[t][id][instanceID]
if a:GetAuraInstanceID() ~= instanceID then if self.playerAuras[id] and self.playerAuras[id][instanceID] then
print("Instance ID mismatch: " .. a:GetAuraInstanceID() .. " " .. instanceID) self.playerAuras[id][instanceID] = nil
end
self[t][id][instanceID] = nil
self.instanceIDLookup[instanceID] = nil self.instanceIDLookup[instanceID] = nil
return
end end
function AuraTable:UpdateInstanceID(instanceID, newAura) if self.auras[id] and self.auras[id][instanceID] then
if not self.instanceIDLookup[instanceID] then self.auras[id][instanceID] = nil
self.instanceIDLookup[instanceID] = nil
return return
end end
end
function AuraTable:AddOrUpdateAuraInstanceID(instanceID, aura)
local spellId = aura:GetSpell():GetID()
self.instanceIDLookup[instanceID] = spellId
local t, id = unpack(self.instanceIDLookup[instanceID]) if Bastion.UnitManager['player']:IsUnit(aura:GetSource()) then
if not self.playerAuras[spellId] then
self.playerAuras[spellId] = {}
end
-- print("Updating aura in table: " .. t .. " " .. id .. " " .. index .. "") self.playerAuras[spellId][instanceID] = aura
else
if not self.auras[spellId] then
self.auras[spellId] = {}
end
self[t][id][instanceID] = newAura self.auras[spellId][instanceID] = aura
end
end end
-- Get a units buffs -- Get a units buffs
function AuraTable:GetUnitBuffs() function AuraTable:GetUnitBuffs()
AuraUtil.ForEachAura(self.unit.unit, 'HELPFUL', nil, function(a) AuraUtil_ForEachAura(self.unit.unit, 'HELPFUL', nil, function(a)
local aura = Bastion.Aura:CreateFromUnitAuraInfo(a) local aura = Bastion.Aura:CreateFromUnitAuraInfo(a)
if aura:IsValid() then if aura:IsValid() then
if aura:GetSource():Exists() and aura:GetSource():IsUnit(Bastion.UnitManager['player']) then self:AddOrUpdateAuraInstanceID(aura:GetAuraInstanceID(), aura)
if not self.playerAppliedBuffs[aura:GetSpell():GetID()] then
self.playerAppliedBuffs[aura:GetSpell():GetID()] = {}
end
self.playerAppliedBuffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'playerAppliedBuffs',
aura:GetSpell():GetID() }
else
if not self.buffs[aura:GetSpell():GetID()] then
self.buffs[aura:GetSpell():GetID()] = {}
end
self.buffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'buffs',
aura:GetSpell():GetID() }
end
end end
end, true) end, true)
end end
-- Get a units debuffs -- Get a units debuffs
function AuraTable:GetUnitDebuffs() function AuraTable:GetUnitDebuffs()
AuraUtil.ForEachAura(self.unit.unit, 'HARMFUL', nil, function(a) AuraUtil_ForEachAura(self.unit.unit, 'HARMFUL', nil, function(a)
local aura = Bastion.Aura:CreateFromUnitAuraInfo(a) local aura = Bastion.Aura:CreateFromUnitAuraInfo(a)
if aura:IsValid() then if aura:IsValid() then
if aura:GetSource():Exists() and aura:GetSource():IsUnit(Bastion.UnitManager['player']) then self:AddOrUpdateAuraInstanceID(aura:GetAuraInstanceID(), aura)
if not self.playerAppliedDebuffs[aura:GetSpell():GetID()] then
self.playerAppliedDebuffs[aura:GetSpell():GetID()] = {}
end
self.playerAppliedDebuffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'playerAppliedDebuffs',
aura:GetSpell():GetID() }
else
if not self.debuffs[aura:GetSpell():GetID()] then
self.debuffs[aura:GetSpell():GetID()] = {}
end
self.debuffs[aura:GetSpell():GetID()][aura:GetAuraInstanceID()] = aura
self.instanceIDLookup[aura:GetAuraInstanceID()] = { 'debuffs',
aura:GetSpell():GetID() }
end
end end
end, true) end, true)
end end
local function merge(t1, t2)
for k, v in pairs(t2) do
if type(v) == "table" then
if type(t1[k] or false) == "table" then
merge(t1[k] or {}, t2[k] or {})
else
t1[k] = v
end
else
t1[k] = v
end
end
return t1
end
-- Update auras -- Update auras
function AuraTable:Update() function AuraTable:Update()
print("Updating auras for " .. tostring(self.unit))
self:Clear() self:Clear()
-- self.lastUpdate = GetTime() -- self.lastUpdate = GetTime()
self:GetUnitBuffs() self:GetUnitBuffs()
self:GetUnitDebuffs() self:GetUnitDebuffs()
-- self.auras = merge(self.buffs, self.debuffs) -- self.auras = self.auras
-- self.playerAuras = merge(self.playerAppliedBuffs, self.playerAppliedDebuffs) -- self.playerAuras = self.playerAuras
end end
-- Get a units auras -- Get a units auras
@ -230,17 +152,17 @@ function AuraTable:GetUnitAuras()
if self.unit:GetGUID() ~= self.guid then if self.unit:GetGUID() ~= self.guid then
self.guid = self.unit:GetGUID() self.guid = self.unit:GetGUID()
self:Update() self:Update()
return merge(self.buffs, self.debuffs) return self.auras
end end
-- -- Cache the auras for the unit so we don't have to query the API every time we want to check if the unit has a specific aura or not -- -- Cache the auras for the unit so we don't have to query the API every time we want to check if the unit has a specific aura or not
-- -- If it's less than .4 seconds since the last time we queried the API, return the cached auras -- -- If it's less than .4 seconds since the last time we queried the API, return the cached auras
-- if self.lastUpdate and GetTime() - self.lastUpdate < 0.5 then -- if self.lastUpdate and GetTime() - self.lastUpdate < 0.5 then
-- return merge(self.buffs, self.debuffs) -- return self.auras
-- end -- end
-- self:Update() -- self:Update()
return merge(self.buffs, self.debuffs) return self.auras
end end
-- Get a units auras -- Get a units auras
@ -253,26 +175,22 @@ function AuraTable:GetMyUnitAuras()
if self.unit:GetGUID() ~= self.guid then if self.unit:GetGUID() ~= self.guid then
self.guid = self.unit:GetGUID() self.guid = self.unit:GetGUID()
self:Update() self:Update()
return merge(self.playerAppliedBuffs, self.playerAppliedDebuffs) return self.playerAuras
end end
-- -- Cache the auras for the unit so we don't have to query the API every time we want to check if the unit has a specific aura or not -- -- Cache the auras for the unit so we don't have to query the API every time we want to check if the unit has a specific aura or not
-- -- If it's less than .4 seconds since the last time we queried the API, return the cached auras -- -- If it's less than .4 seconds since the last time we queried the API, return the cached auras
-- if self.lastUpdate and GetTime() - self.lastUpdate < 0.5 then -- if self.lastUpdate and GetTime() - self.lastUpdate < 0.5 then
-- return merge(self.playerAppliedBuffs, self.playerAppliedDebuffs) -- return self.playerAuras
-- end -- end
-- self:Update() -- self:Update()
return merge(self.playerAppliedBuffs, self.playerAppliedDebuffs) return self.playerAuras
end end
-- Clear the aura table -- Clear the aura table
function AuraTable:Clear() function AuraTable:Clear()
self.buffs = {}
self.debuffs = {}
self.auras = {} self.auras = {}
self.playerAppliedBuffs = {}
self.playerAppliedDebuffs = {}
self.playerAuras = {} self.playerAuras = {}
self.instanceIDLookup = {} self.instanceIDLookup = {}
end end
@ -300,8 +218,7 @@ function AuraTable:Find(spell)
end end
function AuraTable:FindMy(spell) function AuraTable:FindMy(spell)
local auras = self:GetMyUnitAuras() local aurasub = self.playerAuras[spell:GetID()]
local aurasub = auras[spell:GetID()]
if not aurasub then if not aurasub then
return Bastion.Aura:New() return Bastion.Aura:New()

@ -0,0 +1,124 @@
-- Create a NotificationsList class
local NotificationsList = {
notifications = {}
}
NotificationsList.__index = NotificationsList
-- Constructor
function NotificationsList:New()
local self = setmetatable({}, NotificationsList)
-- Create a frame for the notifications
self.frame = CreateFrame("Frame", "BastionNotificationsList", UIParent)
self.frame:SetSize(300, 100)
self.frame:SetPoint("TOP", UIParent, "TOP", 0, -100)
self.frame:SetFrameStrata("HIGH")
-- Remove notifications after 5 seconds
C_Timer.NewTicker(0.1, function()
for i, notification in ipairs(self.notifications) do
if GetTime() - notification.addedAt > 2 then
notification:Remove()
table.remove(self.notifications, i)
end
end
end)
return self
end
-- Create a notification class for the notifications list (takes icon and text)
local Notification = {
}
Notification.__index = Notification
-- Constructor
function Notification:New(list, icon, text)
local self = setmetatable({}, Notification)
-- Create a frame for the notification
self.frame = CreateFrame("Frame", nil, list.frame)
self.frame:SetSize(300, 100)
self.frame:SetPoint("TOP", list.frame, "TOP", 0, 0)
self.frame:SetFrameStrata("HIGH")
-- Create a texture for the icon
self.icon = self.frame:CreateTexture(nil, "ARTWORK")
self.icon:SetSize(32, 32)
self.icon:SetPoint("LEFT", self.frame, "LEFT", 0, 0)
self.icon:SetTexture(icon)
-- Create a fontstring for the text
self.text = self.frame:CreateFontString(nil, "BACKGROUND", "GameFontNormal")
self.text:SetPoint("CENTER", self.frame, "CENTER", 10, 0)
self.text:SetText(text)
self.text:SetFont("Fonts\\FRIZQT__.TTF", 14)
self.addedAt = GetTime()
self.list = list
return self
end
-- Remove notification
function Notification:Remove()
-- Fade out the notification frame and remove it after the fade
UIFrameFadeOut(self.frame, 0.2, 1, 0)
C_Timer.After(0.5, function()
self.frame:Hide()
self.frame:ClearAllPoints()
self.frame:SetParent(nil)
self.frame = nil
self.list:Update()
end)
end
-- Add a notification to the list
function NotificationsList:AddNotification(icon, text)
-- Create a new notification
local notification = Notification:New(self, icon, text)
-- Add the notification to the list
table.insert(self.notifications, notification)
UIFrameFadeIn(notification.frame, 0.2, 0, 1)
-- Update the notifications
self:Update()
end
-- Update the notifications
function NotificationsList:Update()
-- Loop through the notifications
for i, notification in ipairs(self.notifications) do
-- Set the position of the notification
notification.frame:SetPoint("TOP", self.frame, "TOP", 0, -50 * (i - 1))
end
end
-- Remove a notification from the list
function NotificationsList:RemoveNotification(notification)
-- Loop through the notifications
for i, v in ipairs(self.notifications) do
-- Check if the notification is the one we want to remove
if v == notification then
-- Remove the notification from the list
table.remove(self.notifications, i)
notification:Remove()
break
end
end
end
-- Remove all notifications from the list
function NotificationsList:RemoveAllNotifications()
-- Loop through the notifications
for i, v in ipairs(self.notifications) do
-- Remove the notification from the list
table.remove(self.notifications, i)
self.notifications[i]:Remove()
end
end
-- Remove all notifications
return NotificationsList, Notification

@ -40,6 +40,13 @@ function Spell:New(id)
self.spellID = id self.spellID = id
Bastion.EventManager:RegisterWoWEvent("UNIT_SPELLCAST_SUCCEEDED", function(...)
local unit, castGUID, spellID = ...
if unit == "player" and spellID == self:GetID() then
self.lastCastAt = GetTime()
end
end)
return self return self
end end
@ -104,7 +111,7 @@ function Spell:Cast(unit, condition)
-- if unit.unit contains 'nameplate' then we need to use Object wrapper to cast -- if unit.unit contains 'nameplate' then we need to use Object wrapper to cast
local u = unit.unit local u = unit.unit
if string.find(u, 'nameplate') then if type(u) == "string" and string.find(u, 'nameplate') then
u = Object(u) u = Object(u)
end end
@ -114,7 +121,7 @@ function Spell:Cast(unit, condition)
Bastion:Debug("Casting", self) Bastion:Debug("Casting", self)
-- Set the last cast time -- Set the last cast time
self.lastCastAt = GetTime() self.lastCastAttempt = GetTime()
-- Call post cast function -- Call post cast function
if self:GetOnCastFunction() then if self:GetOnCastFunction() then
@ -202,36 +209,20 @@ function Spell:Call(unit)
return false return false
end end
-- Check if the spell is in range of the unit function Spell:HasRange()
function Spell:IsInRange(unit) return SpellHasRange(self:GetName())
local name, rank, icon, castTime, spellmin, spellmax, spellID = GetSpellInfo(self:GetID())
local them = Object(unit.unit)
local tx, ty, tz = ObjectPosition(unit.unit)
local px, py, pz = ObjectPosition('player')
if not them then
return false
end
if tx == 0 and ty == 0 and tz == 0 then
return true
end end
local combatReach = ObjectCombatReach("player") -- Check if the spell is in range of the unit
local themCombatReach = ObjectCombatReach(unit.unit) function Spell:IsInRange(unit)
local hasRange = self:HasRange()
local inRange = IsSpellInRange(self:GetName(), unit.unit)
if Bastion.UnitManager['player']:InMelee(unit) and spellmin == 0 then if not hasRange then
return true return true
end end
local distance = FastDistance(px, py, pz, tx, ty, tz) if hasRange and inRange == 1 then
if spellmax
and distance >= spellmin
and distance <= combatReach + themCombatReach + spellmax
then
return true return true
end end

@ -23,11 +23,15 @@ end
-- tostring -- tostring
function Unit:__tostring() function Unit:__tostring()
return "Bastion.__Unit(" .. self.unit .. ")" .. " - " .. (self:GetName() or '') return "Bastion.__Unit(" .. tostring(self.unit) .. ")" .. " - " .. (self:GetName() or '')
end end
-- Constructor -- Constructor
function Unit:New(unit) function Unit:New(unit)
if UnitIsUnit(unit, "player") then
print("Created unit for player")
end
local self = setmetatable({}, Unit) local self = setmetatable({}, Unit)
self.unit = unit self.unit = unit
self.cache = Bastion.Cache:New() self.cache = Bastion.Cache:New()
@ -144,7 +148,7 @@ end
-- Is the unit a hostile unit -- Is the unit a hostile unit
function Unit:IsHostile() function Unit:IsHostile()
return UnitIsEnemy("player", self.unit) return UnitCanAttack(self.unit, 'player')
end end
-- Is the unit a boss -- Is the unit a boss
@ -217,15 +221,7 @@ local isClassicWow = select(4, GetBuildInfo()) < 40000
-- Check if two units are in melee -- Check if two units are in melee
function Unit:InMelee(unit) function Unit:InMelee(unit)
local combatReach = ObjectCombatReach(self.unit) return UnitInMelee(self.unit, unit.unit)
local themCombatReach = ObjectCombatReach(unit.unit)
if not combatReach or not themCombatReach then
return false
end
return self:GetDistance(unit)
< math.max(combatReach + 1.3333 + themCombatReach, 5)
end end
local losFlag = bit.bor(0x1, 0x10, 0x100000) local losFlag = bit.bor(0x1, 0x10, 0x100000)
@ -317,24 +313,28 @@ function Unit:GetEnemies(range)
end end
local count = 0 local count = 0
local objs = Objects()
local numobjs = #objs Bastion.UnitManager:EnumNameplates(function(unit)
if not self:IsUnit(unit) and unit:GetDistance(self) <= range and unit:IsAlive() and self:CanSee(unit) then
for i = 1, numobjs do
local object = objs[i]
-- unit types 5,6,7
if ObjectType(object) == 5 or ObjectType(object) == 6 then
local unit = Unit:New(object)
if Bastion.UnitManager['player']:CanAttack(unit) and unit:IsAffectingCombat() and unit:IsAlive() and
unit:CanSee(self)
and
self:GetDistance(unit) <= range then
count = count + 1 count = count + 1
end end
end)
self.cache:Set("enemies_" .. range, count, .5)
return count
end end
-- Get the number of melee attackers
function Unit:GetMeleeAttackers()
local count = 0
Bastion.UnitManager:EnumNameplates(function(unit)
if not self:IsUnit(unit) and unit:IsAlive() and self:CanSee(unit) and
self:InMelee(unit) and unit:IsAffectingCombat() and not unit:IsFriendly(self) then
count = count + 1
end end
end)
self.cache:Set("enemies_" .. range, count, .5)
return count return count
end end

@ -69,25 +69,25 @@ function UnitManager:__index(k)
self.cache:Set(k, self.customUnits[k].unit, 0.5) self.cache:Set(k, self.customUnits[k].unit, 0.5)
end end
return self.customUnits[k].unit return self.customUnits[k].unit
end end
if self.objects[k] then local kguid = ObjectGUID(k)
return self.objects[k]
if kguid and self.objects[kguid] then
return self.objects[kguid]
end end
-- if not Validate(k) then -- if not Validate(k) then
-- error("UnitManager:Get - Invalid token: " .. k) -- error("UnitManager:Get - Invalid token: " .. k)
-- end -- end
if self.units[k] == nil then if self.objects[kguid] == nil then
self.units[k] = Unit:New(k) local unit = Unit:New(Object(k))
self:SetObject(unit)
end end
return self.objects[kguid]
return self.units[k]
end end
-- Constructor -- Constructor
@ -109,11 +109,13 @@ function UnitManager:Get(token)
-- error("UnitManager:Get - Invalid token: " .. token) -- error("UnitManager:Get - Invalid token: " .. token)
-- end -- end
if self.units[token] == nil then local tguid = ObjectGUID(token)
self.units[token] = Unit:New(token)
if self.objects[tguid] == nil then
self.objects[tguid] = Unit:New(Object(tguid))
end end
return self.units[token] return self.objects[tguid]
end end
function UnitManager:GetObject(guid) function UnitManager:GetObject(guid)

@ -15,6 +15,7 @@ function Bastion.require(class)
end end
Bastion.ClassMagic = Bastion.require("ClassMagic") Bastion.ClassMagic = Bastion.require("ClassMagic")
Bastion.NotificationsList, Bastion.Notification = Bastion.require("NotificationsList")
Bastion.Vector3 = Bastion.require("Vector3") Bastion.Vector3 = Bastion.require("Vector3")
Bastion.Commmand = Bastion.require("Command") Bastion.Commmand = Bastion.require("Command")
Bastion.Cache = Bastion.require("Cache") Bastion.Cache = Bastion.require("Cache")
@ -34,10 +35,17 @@ Bastion.Class = Bastion.require("Class")
Bastion.Timer = Bastion.require("Timer") Bastion.Timer = Bastion.require("Timer")
Bastion.CombatTimer = Bastion.Timer:New('combat') Bastion.CombatTimer = Bastion.Timer:New('combat')
Bastion.MythicPlusUtils = Bastion.require("MythicPlusUtils"):New() Bastion.MythicPlusUtils = Bastion.require("MythicPlusUtils"):New()
Bastion.Notifications = Bastion.NotificationsList:New()
Bastion.modules = {} Bastion.modules = {}
Bastion.Enabled = false Bastion.Enabled = false
Bastion.EventManager:RegisterWoWEvent('UNIT_AURA', function(unit, auras)
local u = Bastion.UnitManager[unit]
u:GetAuras():OnUpdate(auras)
end)
Bastion.Ticker = C_Timer.NewTicker(0.1, function() Bastion.Ticker = C_Timer.NewTicker(0.1, function()
if not Bastion.CombatTimer:IsRunning() and UnitAffectingCombat("player") then if not Bastion.CombatTimer:IsRunning() and UnitAffectingCombat("player") then
Bastion.CombatTimer:Start() Bastion.CombatTimer:Start()

Loading…
Cancel
Save