/daybreak.lua (520ae9c076f0b7e7e7bfb00557763da51db891c7) (34293 bytes) (mode 100644) (type blob)

--[[--
Daybreak.
Add custom spell activation overlay and paladin holy power bar.
@script daybreak
]]

--[[--
Spell activation overlay.
Custom general purpose spell activation overlay.
@section overlay
]]

--[[--
Access constant default indicator size pixels.
@function getDefaultButtonSize
@treturn number positive integer
]]
local function getDefaultButtonSize()
	return 28
end

--[[--
Process timer tick to update remaining aura duration.
The update employs artificial delay to hopefully reduce the memory cost of updates.
@function applyOverlayUpdate
@tparam frame button button to update
@return nothing
]]
local function applyOverlayUpdate(button)
	assert (button ~= nil)

	local auraName = button.spell
	assert (auraName ~= nil)
	assert ('string' == type(auraName))
	assert (string.len(auraName) >= 2)
	assert (string.len(auraName) <= 256)

	local unitDesignation = button.unit
	assert (unitDesignation ~= nil)
	assert ('string' == type(unitDesignation))
	assert (string.len(unitDesignation) >= 2)
	assert (string.len(unitDesignation) <= 256)

	local filter = button.filter or 'HELPFUL'
	local _, _, _, quantity, _, duration, expirationInstance = UnitAura(unitDesignation, auraName, nil, filter)
	quantity = quantity or 0
	assert (quantity ~= nil)
	assert ('number' == type(quantity))
	quantity = math.ceil(math.min(math.max(0, quantity), 99))

	if nil == expirationInstance then
		return
	end
	local now = GetTime()
	local remainingDuration = math.max(0, math.ceil(expirationInstance - now))
	local isDurationUnlimited = (remainingDuration or 0) == 0 and (duration or 0) == 0
	local t
	if isDurationUnlimited or remainingDuration > 60 then
		t = nil
	elseif quantity < 2 then
		t = tostring(remainingDuration)
	elseif quantity > 9 then
		t = tostring(remainingDuration) .. "*?"
	else
		t = tostring(remainingDuration) .. "*" .. tostring(quantity)
	end
	button:SetText(t)
end

local function overlayUpdateProcessor(self, updateDurationSec)
	assert (self ~= nil)

	assert (updateDurationSec ~= nil)
	assert ('number' == type(updateDurationSec))
	updateDurationSec = math.max(0, updateDurationSec)

	local updateCooldownDurationSec = self.updateCooldownDurationSec
	if nil == updateCooldownDurationSec then
		updateCooldownDurationSec = 0
	end
	assert (updateCooldownDurationSec ~= nil)
	assert ('number' == type(updateCooldownDurationSec))
	updateCooldownDurationSec = math.max(0, updateCooldownDurationSec)
	updateCooldownDurationSec = updateCooldownDurationSec - updateDurationSec

	if updateCooldownDurationSec > 0 then
		self.updateCooldownDurationSec = updateCooldownDurationSec
		return
	else
		updateCooldownDurationSec = 0.084
		self.updateCooldownDurationSec = updateCooldownDurationSec
		applyOverlayUpdate(self)
	end
end

--[[--
Process reaction to UNIT_AURA event for aura indicator.
When aura disappears from the unit associated with this button,
hide the button. Show it otherwise.
@function acceptOverlayUnitAura
@tparam button frame this button frame to update
@tparam string eventCategory given event category designation
@return nothing
]]
local function acceptOverlayUnitAura(button, eventCategory)
	assert (button ~= nil)
	assert (eventCategory ~= nil)
	assert ('string' == type(eventCategory))
	assert (string.len(eventCategory) >= 2)
	assert (string.len(eventCategory) <= 256)

	local auraName = button.spell
	assert (auraName ~= nil)
	assert ('string' == type(auraName))
	assert (string.len(auraName) >= 2)
	assert (string.len(auraName) <= 256)

	local unitDesignation = button.unit
	assert (unitDesignation ~= nil)
	assert ('string' == type(unitDesignation))
	assert (string.len(unitDesignation) >= 2)
	assert (string.len(unitDesignation) <= 256)

	local filter = button.filter or 'HELPFUL'
	local name, _, icon = UnitAura(unitDesignation, auraName, nil, filter)
	if name then
		--[[ FIXME Apply graphics only once instead of every aura update ]]--
		button:SetNormalTexture(icon)
		button:Show()
		button:SetScript('OnUpdate', overlayUpdateProcessor)
	else
		button:Hide()
		button:SetScript('OnUpdate', nil)
	end
end

--[[--
Allocate button frame that represents a single aura
New button has two custom fields: unit and spell.
that is potentially applied to the player character.
The order of parameters allows for predictable
code line sorting.
@function createButton
@tparam number column horizontal position
@tparam number row vertical position
@tparam string localizedSpellName aura name to track
@tparam string buttonName
@tparam frame parentFrame
@treturn frame newly allocated button frame instance
]]
local function createButton(column, row, localizedSpellName, buttonName, parentFrame, filter)
	assert (buttonName ~= nil)
	assert ('string' == type(buttonName))
	assert (string.len(buttonName) >= 2)
	assert (string.len(buttonName) <= 256)

	assert (parentFrame ~= nil)

	assert (localizedSpellName ~= nil)
	assert ('string' == type(localizedSpellName))
	assert (string.len(localizedSpellName) >= 2)
	assert (string.len(localizedSpellName) <= 256)

	local button = CreateFrame('BUTTON', buttonName, parentFrame)
	local padding = 4
	local size = getDefaultButtonSize()
	local s = size + padding
	button:SetSize(size, size)
	button:SetPoint('BOTTOMLEFT', column * s, row * s)

	local text = button:CreateFontString(button:GetName() .. 'Text', 'OVERLAY')
	local fontHandle = DaybreakFont or NumberFont_Outline_Large
	text:SetFontObject(fontHandle)
	text:SetPoint('TOPRIGHT', button, 'TOPRIGHT', 16, 16)
	text:SetPoint('BOTTOMLEfT', button, 'BOTTOMLEFT', -16, -16)

	button:SetFontString(text)
	button:SetText(nil)

	button.spell = localizedSpellName
	button.unit = 'player'
	button.filter = filter or 'HELPFUL'

	button:RegisterEvent('UNIT_AURA')
	button:SetScript('OnEvent', acceptOverlayUnitAura)

	button:Hide()

	return button
end

--[[--
Plyer role assignment.
Assign player role in battlegrounds automatically.
@section role
]]

local function getEstimatedBattlegroundRole(unitDesignation)
	assert (unitDesignation ~= nil)
	assert ('string' == type(unitDesignation))
	assert (string.len(unitDesignation) >= 2)
	assert (string.len(unitDesignation) <= 256)


	--[[ TODO Add complex rolecheck cases for Death Knight and Druid ]]--
	local roleDesignation = 'NONE'
	local _, classDesignation = UnitClass(unitDesignation)
	local i = GetPrimaryTalentTree()
	if 'MAGE' == classDesignation then
		roleDesignation = 'DAMAGER'
	elseif 'HUNTER' == classDesignation then
		roleDesignation = 'DAMAGER'
	elseif 'PALADIN' == classDesignation then
		if 1 == i then
			roleDesignation = 'HEALER'
		elseif 2 == i then
			roleDesignation = 'TANK'
		elseif 3 == i then
			roleDesignation = 'DAMAGER'
		end
	elseif 'PRIEST' == classDesignation then
		if 1 == i then
			roleDesignation = 'HEALER'
		elseif 2 == i then
			roleDesignation = 'HEALER'
		elseif 3 == i then
			roleDesignation = 'DAMAGER'
		end
	elseif 'WARRIOR' == classDesignation then
		if 1 == i then
			roleDesignation = 'DAMAGER'
		elseif 2 == i then
			roleDesignation = 'DAMAGER'
		elseif 3 == i then
			roleDesignation = 'TANK'
		end
	elseif 'WARLOCK' == classDesignation then
		roleDesignation = 'DAMAGER'
	end

	assert (roleDesignation ~= nil)
	assert ('string' == type(roleDesignation))
	assert (string.len(roleDesignation) >= 2)
	assert (string.len(roleDesignation) <= 256)
	return roleDesignation
end


local function roleFrameEventProcessor()
	local unitDesignation = 'player'
	local roleDesignation = getEstimatedBattlegroundRole(unitDesignation)
	UnitSetRole(unitDesignation, roleDesignation)
end

local function initRole(rootFrame)
	assert (rootFrame ~= nil)

	local roleFrame = CreateFrame('FRAME', 'DaybreakRoleFrame', rootFrame)
	roleFrame:SetScript('OnEvent', roleFrameEventProcessor)
	roleFrame:RegisterEvent('PLAYER_ENTERING_BATTLEGROUND')
	roleFrame:RegisterEvent('PLAYER_TALENT_UPDATE')
	roleFrame:RegisterEvent('ZONE_CHANGED_NEW_AREA')

	return roleFrame
end

--[[--
Paladin power bar.
Custom power bar to track paladin holy power resource.
@section powerbar
]]

local function getIndicatorTable(holyPowerIndicatorFrame)
	assert (holyPowerIndicatorFrame ~= nil)

	local t = holyPowerIndicatorFrame.children
	assert (t ~= nil)
	assert ('table' == type(t))
	assert (#t >= 1)
	assert (#t <= 256)

	return t
end

--[[--
Access constant default indicator color.
@function getHolyPowerIndicatorColor
@return red green blue
]]
local function getHolyPowerIndicatorColor()
	return 153 / 255, 0 / 255, 102 / 255
end

--[[--
Access constant default indicator color when out of combat.
@function getHolyPowerIndicatorColor
@return red green blue
]]
local function getHolyPowerIndicatorDecayColor()
	return 102 / 255, 51 / 255, 102 / 255
end

--[[--
Access unit designation that this holy power indicator is associated with.
@function getHolyPowerIndicatorUnitDesignation
@tparam self this holy power indicator frame
@treturn string unit designation
]]
local function getHolyPowerIndicatorUnitDesignation(self)
	assert (self ~= nil)

	local unitDesignation = self.unit
	assert (unitDesignation ~= nil)
	assert ('string' == type(unitDesignation))
	assert (string.len(unitDesignation) >= 2)
	assert (string.len(unitDesignation) <= 256)

	return unitDesignation
end

--[[--
Access constant maximum quantity of holy power points.
@function getMaxHolyPower
@treturn number positive integer
]]
local function getMaxHolyPower()
	return 3
end

--[[--
Query the amount of holy power given unit currently possesses.
@function getUnitHolyPower
@tparam string unitDesignation unit to query
@treturn number positive integer
]]
local function getUnitHolyPower(unitDesignation)
	assert (unitDesignation ~= nil)
	assert ('string' == type(unitDesignation))
	assert (string.len(unitDesignation) >= 2)
	assert (string.len(unitDesignation) <= 256)

	local hppq = UnitPower(unitDesignation, SPELL_POWER_HOLY_POWER)
	assert ('number' == type(hppq))
	assert (hppq >= 0)
	assert (hppq <= 1024)
	hppq = math.abs(math.ceil(hppq))

	local maxQuantity = getMaxHolyPower()
	assert (maxQuantity ~= nil)
	assert (hppq <= maxQuantity)

	return hppq
end

--[[--
Access the time instance when given holy power indicator last detected power change.
@function getDecayInstance
@tparam frame holyPowerIndicatorFrame this frame
@treturn number time instance
]]
local function getDecayInstance(holyPowerIndicatorFrame)
	local decayInstance = holyPowerIndicatorFrame.decayInstance
	if nil == decayInstance then
		decayInstance = 0
	end
	if 'number' ~= type(decayInstance) then
		decayInstance = 0
	end
	decayInstance = math.abs(decayInstance)

	assert (decayInstance ~= nil)
	assert ('number' == type(decayInstance))
	assert (decayInstance >= 0)

	return decayInstance
end

--[[--
Calcualte color that holy power indicator should be assigned with given unit state.
@function produceHolyPowerIndicatorColor
@tparam string unitDesignation unit associated with holy power indicator
@return red green blue coef
]]
local function produceHolyPowerIndicatorColor(unitDesignation)
	assert (unitDesignation ~= nil)

	local combatFlag = 1 == UnitAffectingCombat(unitDesignation) or false
	if combatFlag then
		return getHolyPowerIndicatorColor()
	else
		return getHolyPowerIndicatorDecayColor()
	end
end

--[[--
Calcualte transparency that holy power indicator should be assigned with given unit state.
@function produceHolyPowerIndicatorColor
@tparam string unitDesignation unit associated with holy power indicator
@return alpha coef
]]
local function produceHolyPowerIndicatorTransparency(unitDesignation, decayInstance)
	assert (unitDesignation ~= nil)
	assert (decayInstance ~= nil)

	local combatFlag = 1 == UnitAffectingCombat(unitDesignation) or false
	local a = 1
	if not combatFlag then
		local holyPowerPointLifespan = 10
		local now = GetTime()
		decayInstance = math.min(decayInstance, now)
		local decayDuration = math.min(math.max(0, now - decayInstance), holyPowerPointLifespan)
		a = 1 - decayDuration / holyPowerPointLifespan
	end
	return math.min(math.max(0, a), 1)
end

--[[--
Update holy power indicator appearance.
@function applyHolyPowerIndicatorCombat
]]
local function applyHolyPowerIndicatorCombat(self)
	assert (self ~= nil)

	local unitDesignation = getHolyPowerIndicatorUnitDesignation(self)
	assert (unitDesignation ~= nil)

	local hppq = getUnitHolyPower(unitDesignation)
	if hppq < 1 then
		return
	end
	local t = getIndicatorTable(self)
	assert (t ~= nil)
	--[[ Access last holy power indicator ]]--
	local q = t[hppq]
	--[[ Then render it partially transparent to indicate
	--   that it decays out of combat ]]--
	local i = 0
	local r, g, b = produceHolyPowerIndicatorColor(unitDesignation)
	while (i < hppq - 1) do
		i = i + 1
		local p = t[i]
		p:SetTexture(r, g, b, 1)
	end
	if q then
		local decayInstance = getDecayInstance(self)
		assert (decayInstance ~= nil)
		local a = produceHolyPowerIndicatorTransparency(unitDesignation, decayInstance)
		a = math.min(math.max(0.34, a + 0.34), 1)
		q:SetTexture(r, g, b, a)
	end
end

--[[--
Update holy power indicator last update time instance.
@function acceptHolyPowerIndicatorCombatLogEvent
]]
local function acceptHolyPowerIndicatorCombatLogEvent(holyPowerIndicator, eventCategory, instance,
                                                      combatLogEventCategory, ...)
	assert (eventCategory ~= nil)
	assert (instance ~= nil)

	local unitDesignation = getHolyPowerIndicatorUnitDesignation(holyPowerIndicator) or 'player'
	local unitName = select(7, ...)
	local playerName = UnitName(unitDesignation)
	if 'SPELL_ENERGIZE' == combatLogEventCategory and playerName == unitName then
		holyPowerIndicator.decayInstance = GetTime()
	end
end

--[[--
Process UNIT_POWER event for holy power indicator.
@function acceptHolyPowerIndicatorUnitPower
@tparam frame self this holy power indicator
]]
local function acceptHolyPowerIndicatorUnitPower(self)
	assert (self ~= nil)

	local unitDesignation = getHolyPowerIndicatorUnitDesignation(self)
	assert (unitDesignation ~= nil)

	--[[ Assume the table is sorted appropriately ]]--
	local t = getIndicatorTable(self)
	assert (t ~= nil)

	local maxQuantity = getMaxHolyPower()
	assert (maxQuantity ~= nil)

	--[[ Access the quantity of holy power point property of a given unit ]]--
	local hppq = getUnitHolyPower(unitDesignation)
	assert (hppq ~= nil)

	--[[ Display the indicators according the quantity of unit holy power points ]]--
	local combatFlag = 1 == UnitAffectingCombat(unitDesignation) or false

	local prevCombatFlag = self.combatFlag or false
	local unitLeftCombat = not combatFlag and prevCombatFlag
	if unitLeftCombat then
		self.decayInstance = GetTime()
	end
	local prevUnitPower = self.unitPower or 0
	local powerDecayed = not combatFlag and prevUnitPower ~= hppq and hppq > 0
	if powerDecayed then
		self.decayInstance = GetTime()
	end
	self.unitPower = hppq
	self.combatFlag = combatFlag

	local i = 0
	while (i < hppq) do
		i = i + 1
		local p = t[i]
		assert (p ~= nil)
		p:Show()
	end

	while (i < maxQuantity) do
		i = i + 1
		local p = t[i]
		assert (p ~= nil)
		p:Hide()
	end
end

local function holyPowerIndicatorEventProcessor(holyPowerIndicator, eventCategory, ...)
	if 'COMBAT_LOG_EVENT_UNFILTERED' == eventCategory then
		acceptHolyPowerIndicatorCombatLogEvent(holyPowerIndicator, eventCategory, ...)
	elseif 'UNIT_POWER_FREQUENT' == eventCategory then
		acceptHolyPowerIndicatorUnitPower(holyPowerIndicator, eventCategory, ...)
	end
end

--[[--
Produce frame that tracks and displays given unit holy power.
The new frame possesses custom attribute "unit".
@function createHolyPowerIndicator
@tparam string frameName conventionally unique frame desgination
@tparam frame parentFrame parent frame
@tparam string unitDesignation associated unit designation
@treturn frame newly allocated frame
]]
local function createHolyPowerIndicator(frameName, parentFrame, unitDesignation)
	assert (frameName ~= nil)
	assert ('string' == type(frameName))
	assert (string.len(frameName) >= 2)
	assert (string.len(frameName) <= 256)

	assert (parentFrame ~= nil)

	assert (unitDesignation ~= nil)
	assert ('string' == type(unitDesignation))
	assert (string.len(unitDesignation) >= 2)
	assert (string.len(unitDesignation) <= 256)

	local f = CreateFrame('FRAME', frameName, parentFrame)
	local size = 24
	local padding = 4
	local maxQuantity = getMaxHolyPower()
	f:SetSize(maxQuantity * (size + padding) + 4, size + padding + 4)

	local background = f:CreateTexture(frameName .. 'Background', 'BACKGROUND')
	background:SetAllPoints()
	background:SetTexture(0, 0, 0, 0.64)
	f.background = background

	local t = {}
	local q = 0
	local x = 0
	local y = 0
	while (q < maxQuantity) do
		q = q + 1
		local p = f:CreateTexture(frameName .. tostring(q), 'OVERLAY')
		p:SetTexture(getHolyPowerIndicatorColor())
		p:SetSize(size, size)
		p:SetPoint('BOTTOMLEFT', padding + x * (size + padding), y + padding)
		p:Hide()
		x = x + 1
		t[q] = p
	end

	f.unit = unitDesignation
	f.children = t

	f:RegisterEvent('COMBAT_LOG_EVENT_UNFILTERED')
	f:RegisterEvent('UNIT_POWER_FREQUENT')
	f:SetScript('OnEvent', holyPowerIndicatorEventProcessor)
	f:SetScript('OnUpdate', applyHolyPowerIndicatorCombat)
	return f
end

local function initPowerBar(rootFrame)
	assert (rootFrame ~= nil)

	local _, c = UnitClass('player')
	if 'PALADIN' ~= c then
		return
	end

	local hpf = createHolyPowerIndicator('DaybreakHolyPowerPlayerFrame', rootFrame, 'player')
	hpf:SetPoint('CENTER', 0, 0)
	hpf:SetPoint('BOTTOM', 0, 0)

	PaladinPowerBar:Hide()
	PaladinPowerBar:SetScript('OnEvent', nil)
	PaladinPowerBar:SetScript('OnLoad', nil)
	PaladinPowerBarBG:Hide()

	return hpf
end

--[[--
Lose control indicator.
@section cyclone
]]

local function createCycloneButton(column, row, localizedSpellName, buttonDesignation, parentFrame)
	return createButton(column, row, localizedSpellName, buttonDesignation, parentFrame, 'HARMFUL')
end

--[[--
Initialize character lose control indicators.
@function initCyclone
@tparam frame rootFrame parent frame
@treturn frame newly allocated frame
]]
local function initCyclone(rootFrame)
	--[[ Row: lose control category; column: debuff type (magic, poison, phys) ]]--
	local p = CreateFrame('FRAME', 'DaybreakCycloneHeader', rootFrame)
	p:SetSize(32 * 4, 32 * 4)
	p:SetPoint('CENTER', 0, 0)

	--[[ Ass ]]--
	createCycloneButton(2, 4, 'Cyclone', 'DaybreakCycloneCyclone', p)

	--[[ Stun ]]--
	createCycloneButton(2, 1, 'Charge', 'DaybreakCycloneCharge', p)
	createCycloneButton(2, 1, 'Cheap Shot', 'DaybreakCycloneCheapShot', p)
	createCycloneButton(2, 1, 'Concussion Blow', 'DaybreakCycloneConcussionBlow', p)
	createCycloneButton(2, 1, 'Fire Blast', 'DaybreakCycloneFireBlast', p)
	createCycloneButton(2, 1, 'Gnaw', 'DaybreakCycloneGnaw', p)
	createCycloneButton(2, 1, 'Intercept', 'DaybreakCycloneIntercept', p)
	createCycloneButton(2, 1, 'Kidney Shot', 'DaybreakCycloneKidenyShot', p)
	createCycloneButton(2, 1, 'Shockwave', 'DaybreakCycloneShockwave', p)
	createCycloneButton(2, 1, 'Throwdown', 'DaybreakCycloneThrowdown', p)
	createCycloneButton(2, 4, 'Deep Freeze', 'DaybreakCycloneDeepFreeze', p)
	createCycloneButton(2, 4, 'Hammer of Justice', 'DaybreakCycloneHammerOfJustice', p)

	--[[ Disorient ]]--
	createCycloneButton(1, 1, 'Blind', 'DaybreakCycloneBlind', p)
	createCycloneButton(1, 1, 'Gouge', 'DaybreakCycloneGouge', p)
	createCycloneButton(1, 1, 'Sap', 'DaybreakCycloneSap', p)
	createCycloneButton(1, 2, 'Hex', 'DaybreakCycloneHex', p)
	createCycloneButton(1, 4, 'Death Coil', 'DaybreakCycloneFear', p)
	createCycloneButton(1, 4, 'Dragon Breath', 'DaybreakCycloneDragonBreath', p)
	createCycloneButton(1, 4, 'Fear', 'DaybreakCycloneFear', p)
	createCycloneButton(1, 4, 'Feezing Trap', 'DaybreakCycloneFreezingTrap', p)
	createCycloneButton(1, 4, 'Holy Word: Chastise', 'DaybreakCycloneHolyWordChastise', p)
	createCycloneButton(1, 4, 'Howl of Terror', 'DaybreakCycloneFear', p)
	createCycloneButton(1, 4, 'Intimidating Shout', 'DaybreakCycloneIntimidatingShout', p)
	createCycloneButton(1, 4, 'Polymorph', 'DaybreakCycloneFear', p)
	createCycloneButton(1, 4, 'Ring of Frost', 'DaybreakCycloneRingOfFrost', p)

	--[[ Silence ]]--
	createCycloneButton(1, 1, 'Gag Order', 'DaybreakCycloneGagOrder', p)
	createCycloneButton(1, 4, 'Counterspell', 'DaybreakCycloneCounterspell', p)
	createCycloneButton(1, 4, 'Silence', 'DaybreakCycloneSilence', p)
	createCycloneButton(1, 4, 'Spell Lock', 'DaybreakCycloneSpellLock', p)
	createCycloneButton(1, 4, 'Strangulate', 'DaybreakCycloneFear', p)

	--[[ Root ]]--
	createCycloneButton(0, 4, 'Frost Nova', 'DaybreakCycloneFrostNova', p)
	createCycloneButton(0, 4, 'Entangling Roots', 'DaybreakCycloneEntanglingRoots', p)

	--[[ Slow ]]--
	createCycloneButton(0, 1, 'Chains of Ice', 'DaybreakCycloneChainsOfIce', p)
	createCycloneButton(0, 1, 'Hamstring', 'DaybreakCycloneHamstring', p)
	createCycloneButton(0, 3, 'Crippling Poison', 'DaybreakCycloneCripplingPoison', p)
	createCycloneButton(0, 4, 'Cone of Cold', 'DaybreakCycloneConeOfCold', p)
	createCycloneButton(0, 4, 'Earthbind Totem', 'DaybreakCycloneEarthbindTotem', p)
	createCycloneButton(0, 4, 'Slow', 'DaybreakCycloneSlow', p)

	--[[ Healing reduction ]]--
	createCycloneButton(3, 1, 'Aimed Shot', 'DaybreakCycloneAimedShot', p)
	createCycloneButton(3, 1, 'Mortal Strike', 'DaybreakCycloneMortalStrike', p)
	createCycloneButton(3, 1, 'Necrotic Strike', 'DaybreakCycloneNecroticStrike', p)
	createCycloneButton(3, 3, 'Wound Poison', 'DaybreakCycloneWoundPoison', p)
	createCycloneButton(3, 2, 'Curse of Tongues', 'DaybreakCycloneCurseOfTongues', p)
end

--[[--
Beacon of light.
Add dedicated Beacon of Light indicator.
@section beacon
]]

local function beaconEventProcessor(self, ...)
	local combatLogEventCategory = select(3, ...)
	local unitName1 = select(6, ...)
	local unitName2 = select(10, ...)
	local spellName = select(14, ...)
	--[[print(combatLogEventCategory, unitName1, unitName2, spellName)]]--
	local playerName = UnitName('player')
	if playerName ~= unitName1 then
		return
	end
	if 'Beacon of Light' ~= spellName then
		return
	end
	if 'SPELL_AURA_APPLIED' == combatLogEventCategory or
	   'SPELL_AURA_REFRESH' == combatLogEventCategory then
		self.unitName = unitName2
		self:Show()
	end
	if 'SPELL_AURA_REMOVED' == combatLogEventCategory then
		self.unitName = nil
		self:Hide()
	end
end

local function beaconUpdateProcessor(self)
	if nil == self then
		return
	end
	local t = {
		'party1', 'party2', 'party3', 'party4',
		'raid1', 'raid2', 'raid3', 'raid4', 'raid5',
		'raid5', 'raid6', 'raid7', 'raid8', 'raid9',
		'raid10', 'raid11', 'raid12', 'raid13', 'raid14',
		'raid15', 'raid16', 'raid17', 'raid18', 'raid19',
		'raid20', 'raid21', 'raid22', 'raid23', 'raid24',
		'raid25', 'raid26', 'raid27', 'raid28', 'raid29',
		'raid30', 'raid31', 'raid32', 'raid33', 'raid34',
		'raid35', 'raid36', 'raid37', 'raid38', 'raid39',
		'player'
	}
	local m = self.unitName
	if nil == m then
		self:Hide()
		return
	end
	local d
	local i = 0
	while (i < #t) do
		i = i + 1
		local r = t[i]
		local n = UnitName(r)
		if n == m then
			d = r
			break
		end
	end
	if nil == d then
		self.unitName = nil
		self:Hide()
		return
	end
	if UnitIsDead(d) then
		self.unitName = nil
		self:Hide()
	end

	local artwork = self:GetNormalTexture()
	if 1 == IsSpellInRange('Beacon of Light', d) then
		artwork:SetVertexColor(1, 1, 1, 1)
	else
		artwork:SetVertexColor(1, 1, 1, 0.4)
	end
end

--[[--
Initialize player Beacon of Light buff indicator.
@function initBeacon
@tparam frame rootFrame parent frame
@treturn frame new frame that tracks the specific player buff
--]]
local function initBeacon(rootFrame)
	local beacon = createButton(11, 4, 'Beacon of Light', 'DaybreakOverlayPaladinBeaconOfLight', rootFrame)
	beacon:SetNormalTexture("Interface\\Icons\\Ability_Paladin_BeaconOfLight")

	beacon:UnregisterAllEvents()
	beacon:SetScript('OnEvent', beaconEventProcessor)
	beacon:SetScript('OnUpdate', beaconUpdateProcessor)
	beacon:RegisterEvent('COMBAT_LOG_EVENT_UNFILTERED')

	return beacon
end

--[[--
Blessed life.
Track availability of Blessed Life effect on the player character.
@section blessedlife
]]

--[[--
Query if the player character invested into Blessed Life talent.
@function isBlessedLifePresent
@treturn bool
]]
local function isBlessedLifePresent()
	local specNumber = 1
	local talentNumber = 19
	local _, _, _, _, a = GetTalentInfo(specNumber, talentNumber)
	if nil == a then
		return false
	end
	if 'number' ~= type(a) then
		return false
	end
	return a >= 1
end

--[[--
Process combat log events to track Blessed Life effect.
See COMBAT_LOG_EVENT event category.
@function acceptBlessedLifeCombatLogEvent
@tparam frame blessedLife self
]]
local function acceptBlessedLifeCombatLogEvent(blessedLife, eventCategory, instance,
                                               combatLogEventCategory, ...)
	assert (blessedLife ~= nil)
	assert (eventCategory ~= nil)
	assert (instance ~= nil)

	if not blessedLife.isEnabled then
		return
	end

	local now = GetTime()
	local unitDesignation = 'player'

	--[[ Process if blessed life effect occurred
	--   if it did, remember the time instance ]]--
	local unitName = select(7, ...)
	local playerName = UnitName(unitDesignation)
	if 'SPELL_ENERGIZE' == combatLogEventCategory and playerName == unitName then
		local spellName = select(11, ...)
		if 'Blessed Life' == spellName then
			blessedLife.lastEffectInstance = now
		end
	end

	--[[ Calculate amount of seconds passed since last proc ]]--
	local blessedLifeCooldown = 8
	local lastEffectInstance = blessedLife.lastEffectInstance or 0
	local blessedLifeReady = math.abs(now - lastEffectInstance) > blessedLifeCooldown

	--[[ If player is in combat and more than eight seconds passed since last proc
	--   then show that blessed life is ready to proc ]]--
	local combatFlag = 1 == UnitAffectingCombat(unitDesignation) or false
	if combatFlag and blessedLifeReady then
		blessedLife:Show()
	else
		blessedLife:Hide()
	end
end

--[[--
Detect if Blessed Life ability is still available after spell set change.
@function acceptBlessedLifeSpellsChanged
]]
local function acceptBlessedLifeSpellsChanged(blessedLife)
	assert (blessedLife ~= nil)
	blessedLife.isEnabled = isBlessedLifePresent()
end

--[[--
Route events that concern Blessed Life indicator.
@function blessedLifeEventProcessor
]]
local function blessedLifeEventProcessor(blessedLife, eventCategory, ...)
	if 'COMBAT_LOG_EVENT_UNFILTERED' == eventCategory then
		acceptBlessedLifeCombatLogEvent(blessedLife, eventCategory, ...)
	elseif 'PLAYER_TALENT_UPDATE' == eventCategory then
		acceptBlessedLifeSpellsChanged(blessedLife, eventCategory, ...)
	elseif 'SPELLS_CHANGED' == eventCategory then
		acceptBlessedLifeSpellsChanged(blessedLife, eventCategory, ...)
	end
end

--[[--
Create an indicator for Blessed Life effect for the player character.
@function initBlessedLife
@tparam frame rootFrame
@treturn frame
]]
local function initBlessedLife(rootFrame)
	assert (rootFrame ~= nil)

	local blessedLife = createButton(2, 0, 'Blessed Life', 'DaybreakOverlayPaladinBlessedLife', rootFrame)
	blessedLife:SetNormalTexture("Interface\\Icons\\Spell_holy_blessedlife")

	blessedLife:UnregisterAllEvents()
	blessedLife.isEnabled = isBlessedLifePresent()
	blessedLife:SetScript('OnEvent', blessedLifeEventProcessor)
	blessedLife:RegisterEvent('COMBAT_LOG_EVENT_UNFILTERED')
	blessedLife:RegisterEvent('PLAYER_TALENT_UPDATE')
	blessedLife:RegisterEvent('SPELLS_CHANGED')

	blessedLife:Hide()

	return blessedLife
end

--[[--
Priest overlay.
Priest spell activation overlay.
@section serendipity
]]

--[[--
Initialize priest spell activation overlay if necessary.
Must only be called once per script lifetime.
Assumes that player class is constant.
@function initSerendipity
@tparam frame rootFrame addon root frame to parent indicators from
@return void
]]
local function initSerendipity(rootFrame)
	assert (rootFrame ~= nil)

	local _, classDesignation = UnitClass('player')
	if 'PRIEST' ~= classDesignation then
		return
	end

	--[[ Priest ]]--
	createButton(0, 0, 'Chakra: Chastise', 'DaybreakOverlayPriestChakraChastise', rootFrame)
	createButton(0, 0, 'Chakra: Sanctuary', 'DaybreakOverlayPriestChakraSanctuary', rootFrame)
	createButton(0, 0, 'Chakra: Serenity', 'DaybreakOverlayPriestChakraSerenity', rootFrame)
	createButton(0, 1, 'Chakra', 'DaybreakOverlayPriestChakra', rootFrame)
	--[[ Holy ]]--
	createButton(1, 0, 'Holy Word: Serenity', 'DaybreaksOverlayPriestHolyWordSerenity', rootFrame)
	createButton(1, 1, 'Surge of Light', 'DaybreakOverlayPriestSurgeOfLight', rootFrame)
	createButton(1, 2, 'Serendipity', 'DaybreakOverlayPriestSerendipity', rootFrame)
	createButton(1, 3, 'Guardian Spirit', 'DaybreakOverlayPriestGuardianSpirit', rootFrame)

	--[[ Priest general ]]--
	createButton(9, 0, 'Inspiration', 'DaybreakOverlayPriestInspiration', rootFrame)
	createButton(9, 1, 'Renew', 'DaybreakOverlayPriestRenew', rootFrame)
	createButton(9, 2, 'Power Word: Shield', 'DaybreakOverlayPriestPowerWordShield', rootFrame)
	createButton(10, 2, 'Body and Soul', 'DaybreakOverlayPriestBodyAndSoul', rootFrame)

	--[[ Hide native spell proc indicators that flash in the middle of the screen ]]--
	SetCVar("displaySpellActivationOverlays", false)
end

--[[--
Paladin overlay.
Paladin spell activation overlay.
@section daybreak
]]

--[[--
Initialize paladin spell activation overlay if necessary.
Must only be called once per script lifetime.
Assumes that player class is constant.
@function initDaybreak
@tparam frame rootFrame addon root frame to parent indicators from
@return void
]]
local function initDaybreak(rootFrame)
	assert (rootFrame ~= nil)

	local _, classDesignation = UnitClass('player')
	if 'PALADIN' ~= classDesignation then
		return
	end

	--[[ Paladin ]]--
	--[[ General ]]--
	createButton(0, 0, 'Crusader', 'DaybreakOverlayPaladinCrusader', rootFrame)
	createButton(0, 1, 'Divine Plea', 'DaybreakOverlayPaladinDivinePlea', rootFrame)
	createButton(0, 2, 'Divine Protection', 'DaybreakOverlayPaladinDivineProtection', rootFrame)
	createButton(0, 2, 'Divine Shield', 'DaybreakOverlayPaladinDivineShield', rootFrame)
	createButton(0, 3, 'Avenging Wrath', 'DaybreakOverlayPaladinAvengingWrath', rootFrame)
	createButton(0, 4, 'Guardian of Ancient Kings', 'DaybreakOverlayPaladinGuardianOfTheAncientKings', rootFrame)

	createButton(11, 0, 'Concentration Aura', 'DaybreakOverlayPaladinConcentrationAura', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 0, 'Crusader Aura', 'DaybreakOverlayPaladinCrusaderAura', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 0, 'Devotion Aura', 'DaybreakOverlayPaladinDevotionAura', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 0, 'Resistance Aura', 'DaybreakOverlayPaladinResistanceAura', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 0, 'Retribution Aura', 'DaybreakOverlayPaladinRetributionAura', rootFrame, 'PLAYER HELPFUL')

	createButton(11, 1, 'Seal of Insight', 'DaybreakOverlayPaladinSealOfInsight', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 1, 'Seal of Justice', 'DaybreakOverlayPaladinSealOfJustice', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 1, 'Seal of Righteousness', 'DaybreakOverlayPaladinSealOfRighteousness', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 1, 'Seal of Truth', 'DaybreakOverlayPaladinSealOfTruth', rootFrame, 'PLAYER HELPFUL')

	createButton(11, 2, 'Blessing of Might', 'DaybreakOverlayPaladinBlessingOfMight', rootFrame, 'PLAYER HELPFUL')
	createButton(11, 2, 'Blessing of Kings', 'DaybreakOverlayPaladinBlessingOfKings', rootFrame, 'PLAYER HELPFUL')

	--[[ Effects that may be applied by other players place on the right side ]]--
	createButton(9, 0, 'Hand of Freedom', 'DaybreakOverlayPaladinHandOfFreedom', rootFrame)
	createButton(9, 1, 'Hand of Sacrifice', 'DaybreakOverlayPaladinHandOfSacrifice', rootFrame)
	createButton(9, 2, 'Hand of Protection', 'DaybreakOverlayPaladinHandOfProtection', rootFrame)
	createButton(9, 3, 'Divine Sacrifice', 'DaybreakOverlayPaladinDivineSacrifice', rootFrame)
	createButton(9, 4, 'Aura Mastery', 'DaybreakOverlayPaladinAuraMastery', rootFrame)
	createButton(10, 0, 'Illuminated Healing', 'DaybreakOverlayPaladinIlluminatedHealing', rootFrame)
	createButton(10, 1, 'Conviction', 'DaybreakOverlayPaladinConviction', rootFrame)
	createButton(10, 2, 'Power Torrent', 'DaybreakOverlayPaladinPowerTorrent', rootFrame)
	createButton(10, 3, 'Surge of Dominance', 'DaybreakOverlayPaladinSurgeOfDominance', rootFrame)

	--[[ Holy ]]--
	createButton(1, 0, 'Judgements of the Pure', 'DaybreakOverlayPaladinJudgementsOfThePure', rootFrame)
	createButton(1, 2, 'Daybreak', 'DaybreakOverlayPaladinDaybreak', rootFrame)
	createButton(1, 3, 'Infusion of Light', 'DaybreakOverlayPaladinInfusionOfLight', rootFrame)
	createButton(1, 4, 'Divine Favor', 'DaybreakOverlayPaladinDivineFavor', rootFrame)

	--[[ Protection ]]--
	createButton(1, 0, 'Guarded by the Light', 'DaybreakOverlayPaladinGuardedByTheLight', rootFrame)
	createButton(1, 1, 'Holy Shield', 'DaybreakOverlayPaladinHolyShield', rootFrame)
	createButton(1, 2, 'Ardent Defender', 'DaybreakOverlayPaladinArdentDefender', rootFrame)
	createButton(1, 3, 'Grand Crusader', 'DaybreakOverlayPaladinGrandCrusader', rootFrame)
	createButton(1, 4, 'Sacred Duty', 'DaybreakOverlayPaladinSacredDuty', rootFrame)

	--[[ Retribution ]]--
	createButton(10, 4, 'Inquisition', 'DaybreakOverlayPaladinInquisition', rootFrame)

	--[[ Hide native spell proc indicators that flash in the middle of the screen ]]--
	SetCVar("displaySpellActivationOverlays", false)
end

--[[--
When variables loaded then create and configure all required frames.
Must only be executed once per script lifetime.
@function init
@tparam frame rootFrame
@treturn frame updated given root frame
]]
local function init(rootFrame)
	assert (rootFrame ~= nil)

	rootFrame:SetSize(384, 256)
	rootFrame:SetPoint('CENTER', UIParent,
	                   'CENTER', 0, 0)

	initCyclone(rootFrame)
	initBeacon(rootFrame)
	initBlessedLife(rootFrame)
	initDaybreak(rootFrame)
	initPowerBar(rootFrame)
	initRole(rootFrame)
	initSerendipity(rootFrame)

	print('[Daybreak]: Addon loaded.')

	return rootFrame
end

--[[--
Daybreak script entry point.
Must only be executed once per script life time.
@function main
]]
local function main()
	local rootFrame = CreateFrame('FRAME', 'DaybreakFrame', UIParent)
	rootFrame:SetScript('OnEvent', init)
	rootFrame:RegisterEvent('VARIABLES_LOADED')
end
main()


Mode Type Size Ref File
100644 blob 13 15bf5509b89f302c3b1f5791ce05b051e20b3b6c .gitignore
100644 blob 2012 455ea24126c74d7786c82c34dcf0e10942f2fbc9 .luacheckrc
100644 blob 34293 520ae9c076f0b7e7e7bfb00557763da51db891c7 daybreak.lua
100644 blob 158 463fd022eac55b50d0088bd1b0a27dcbcec2e5b1 daybreak.toc
100644 blob 467 2df405be5cdce8e0151e4fee7d0e9fd7f8c49f67 daybreak.xml
100644 blob 12279720 fa0fefa2a615dfdc182c149a6232b091430171c2 unifont.ttf
Hints:
Before first commit, do not forget to setup your git environment:
git config --global user.name "your_name_here"
git config --global user.email "your@email_here"

Clone this repository using HTTP(S):
git clone https://rocketgit.com/user/vrtc/wowaddons

Clone this repository using ssh (do not forget to upload a key first):
git clone ssh://rocketgit@ssh.rocketgit.com/user/vrtc/wowaddons

Clone this repository using git:
git clone git://git.rocketgit.com/user/vrtc/wowaddons

You are allowed to anonymously push to this repository.
This means that your pushed commits will automatically be transformed into a merge request:
... clone the repository ...
... make some changes and some commits ...
git push origin main