Grichelde bootstrap
parent
0717673624
commit
6ff629284a
@ -0,0 +1,43 @@
|
|||||||
|
# Gradle
|
||||||
|
.gradle
|
||||||
|
build/
|
||||||
|
|
||||||
|
# Android built artifacts
|
||||||
|
*.apk
|
||||||
|
*.ap_
|
||||||
|
*.dex
|
||||||
|
|
||||||
|
# Java build artifacts class files
|
||||||
|
*.class
|
||||||
|
|
||||||
|
# other generated files
|
||||||
|
gen/
|
||||||
|
target/
|
||||||
|
|
||||||
|
# local configuration file (for Android sdk path, etc)
|
||||||
|
local.properties
|
||||||
|
|
||||||
|
# OSX files
|
||||||
|
.DS_Store
|
||||||
|
|
||||||
|
# Eclipse
|
||||||
|
/.classpath
|
||||||
|
/.settings/
|
||||||
|
/.project
|
||||||
|
/.metadata
|
||||||
|
bin/
|
||||||
|
|
||||||
|
# IntelliJ
|
||||||
|
*.iws
|
||||||
|
*.uml
|
||||||
|
.idea/
|
||||||
|
out/
|
||||||
|
|
||||||
|
# NDK
|
||||||
|
obj/
|
||||||
|
|
||||||
|
# Misc
|
||||||
|
twitch/
|
||||||
|
*.log
|
||||||
|
*.graphml
|
||||||
|
coverage.db*
|
@ -0,0 +1,2 @@
|
|||||||
|
# Grichelde
|
||||||
|
|
@ -0,0 +1,500 @@
|
|||||||
|
--[[--------------------------------------------------------------------------
|
||||||
|
Grichelde
|
||||||
|
Copyright 2020 Teilzeit-Jedi <tj@teilzeit-jedi.de>
|
||||||
|
|
||||||
|
based on Misspelled developed by Nathan Pieper - nrpieper (@) gmail (dot) com
|
||||||
|
|
||||||
|
This code freely distributed for your use in any GPL compliant project.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------]] --
|
||||||
|
|
||||||
|
local Grichelde = {}
|
||||||
|
Grichelde = LibStub("AceAddon-3.0"):NewAddon("Grichelde", "AceEvent-3.0", "AceHook-3.0")
|
||||||
|
Grichelde.Version = "0.1.0"
|
||||||
|
|
||||||
|
local AceGUI = LibStub("AceGUI-3.0")
|
||||||
|
local L = LibStub("AceLocale-3.0"):GetLocale("Grichelde", true)
|
||||||
|
|
||||||
|
-- faster function lookups by mapping to local refs
|
||||||
|
local string_find = string.find
|
||||||
|
local string_gsub = string.gsub
|
||||||
|
local string_len = string.len
|
||||||
|
local string_rep = string.rep
|
||||||
|
local string_sub = string.sub
|
||||||
|
local tostring = string.tostring
|
||||||
|
local tContains = tContains
|
||||||
|
local strtrim = strtrim
|
||||||
|
local pairs = table.pairs
|
||||||
|
local ipairs = table.ipairs
|
||||||
|
|
||||||
|
local Grichelde_Saved_CTL_SendChatMessage
|
||||||
|
local Grichelde_CTL_hookedversion = 0
|
||||||
|
|
||||||
|
local Grichelde_ChatTypes = { "SAY", "EMOTE", "YELL", "PARTY", "GUILD", "OFFICER", "RAID", "RAID_WARNING", "INSTANCE_CHAT", "BATTLEGROUND", "WHISPER" }
|
||||||
|
local Grichelde_ChatCommands = { "/s", "/e", "/me", "/y", "/p", "/pl", "/g", "/o", "/raid", "/rl", "/rw", "/i", "bg", "/w", "/r", "/tt" }
|
||||||
|
|
||||||
|
-- do not replace these patterns
|
||||||
|
local Grichelde_IgnorePatterns = {
|
||||||
|
"|[Cc]%x%x%x%x%x%x%x%x.-|r", -- colored items (or links)
|
||||||
|
"|H.-|h", -- item links (http://www.wowwiki.com/ItemLink)
|
||||||
|
"|T.-|t", -- textures
|
||||||
|
"|n", -- newline
|
||||||
|
"{rt[1-8]}", -- rumbered raid target icons
|
||||||
|
"{Star}", -- named raid target icon 1
|
||||||
|
"{Circle}", -- named raid target icon 2
|
||||||
|
"{Coin}", -- named raid target icon 2
|
||||||
|
"{Diamond}", -- named raid target icon 3
|
||||||
|
"{Triangle}", -- named raid target icon 4
|
||||||
|
"{Moon}", -- named raid target icon 5
|
||||||
|
"{Square}", -- named raid target icon 6
|
||||||
|
"{Cross}", -- named raid target icon 7
|
||||||
|
"{X}", -- named raid target icon 7
|
||||||
|
"{Skull}" -- named raid target icon 8
|
||||||
|
}
|
||||||
|
|
||||||
|
function Grichelde:OnInitialize()
|
||||||
|
-- print("Grichelde was loaded")
|
||||||
|
|
||||||
|
-- Build Interface Options window
|
||||||
|
self:CreateInterfaceOptions()
|
||||||
|
|
||||||
|
-- Watch for WIM and Prat to Load, then integrate
|
||||||
|
Grichelde:RegisterEvent("ADDON_LOADED")
|
||||||
|
|
||||||
|
-- hooks for removing any misspelled word highlighting in the text before the chat message is sent
|
||||||
|
-- The WoW client will disconnect if you attempt to send a color tags in a chat message.
|
||||||
|
Grichelde:RawHook("SendChatMessage", true)
|
||||||
|
end
|
||||||
|
|
||||||
|
--- @param event string
|
||||||
|
--- @param addonName string
|
||||||
|
function Grichelde:ADDON_LOADED(event, addonName)
|
||||||
|
if event == "ADDON_LOADED" and addonName == "WIM" then
|
||||||
|
WIM.RegisterWidgetTrigger("msg_box", "whisper,chat,w2w", "OnEnterPressed", Grichelde.EditBox_OnEnterPressed)
|
||||||
|
|
||||||
|
-- WIM sends its chat messages via the API ChatThrottleLib, which itself hooks the default SendChatMessage api
|
||||||
|
-- many times before Grichelde will. ChatThrottleLib might potentially load before Grichelde, so we just hook
|
||||||
|
-- into ChatThrottleLib to be on the safe side.
|
||||||
|
|
||||||
|
if (WIM.RegisterPreSendFilterText) then -- avoid error if WIM not up to date.
|
||||||
|
WIM.RegisterPreSendFilterText(function(text)
|
||||||
|
return Grichelde:ReplaceText(text)
|
||||||
|
end)
|
||||||
|
else
|
||||||
|
if (ChatThrottleLib and Grichelde_CTL_hookedversion < ChatThrottleLib.version) then
|
||||||
|
Grichelde_Saved_CTL_SendChatMessage = ChatThrottleLib.SendChatMessage
|
||||||
|
|
||||||
|
function ChatThrottleLib:SendChatMessage(prio, prefix, text, ...)
|
||||||
|
text = Grichelde:ReplaceText(text)
|
||||||
|
-- print("Grichelde Hooked ChatThrottleLib_SendChatMessaged called")
|
||||||
|
return Grichelde_Saved_CTL_SendChatMessage(ChatThrottleLib, prio, prefix, text, ...)
|
||||||
|
end
|
||||||
|
|
||||||
|
Grichelde_CTL_hookedversion = ChatThrottleLib.version
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Before a chat message is sent, check if replacement is required and replace the text accordingly.
|
||||||
|
--- @param message string
|
||||||
|
--- @param type string
|
||||||
|
--- @param language string
|
||||||
|
--- @param channel string
|
||||||
|
function Grichelde:SendChatMessage(message, type, language, channel, ...)
|
||||||
|
local text = strtrim(message)
|
||||||
|
|
||||||
|
if (Grichelde:CheckReplacement(message, type)) then
|
||||||
|
text = Grichelde:ReplaceText(text)
|
||||||
|
end
|
||||||
|
|
||||||
|
self.hooks["SendChatMessage"](text, type, language, channel, ...);
|
||||||
|
end
|
||||||
|
|
||||||
|
function Grichelde:CheckReplacement(text, type)
|
||||||
|
-- todo: globally disabled?
|
||||||
|
|
||||||
|
-- check type
|
||||||
|
if (not tContains(Grichelde_ChatTypes, type)) then
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
-- don't replace slash commands except chat related commands
|
||||||
|
if string_sub(text, 1, 1) == "/" then
|
||||||
|
local firstWord, _ = Grichelde:GetNextWord(text)
|
||||||
|
if (firstWord == nil or not tContains(Grichelde_ChatCommands, firstWord)) then
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Replaces all character occurrences for which replacements have been defined in the options,
|
||||||
|
--- while preserving any itemLinks or textures. (http://www.wowwiki.com/ItemLink)
|
||||||
|
--- @param text string
|
||||||
|
--- @return string
|
||||||
|
function Grichelde:ReplaceText(text)
|
||||||
|
local finalText = ""
|
||||||
|
local newText = text
|
||||||
|
|
||||||
|
-- don't replace non-chat related slash commands
|
||||||
|
local firstWord, line = Grichelde:GetNextWord(text)
|
||||||
|
if (tContains(Grichelde_ChatCommands, firstWord)) then
|
||||||
|
-- skip chat slash command
|
||||||
|
finalText = finalText .. firstWord .. ' '
|
||||||
|
newText = line
|
||||||
|
end
|
||||||
|
|
||||||
|
local current = 1
|
||||||
|
local lastStart = 1
|
||||||
|
|
||||||
|
while current < string_len(newText) do
|
||||||
|
local currentChar = string_sub(newText, current, current)
|
||||||
|
if currentChar ~= '|' and currentChar ~= '{' then
|
||||||
|
current = current + 1
|
||||||
|
else
|
||||||
|
-- lookahead-check for itemLinks, textures and raid target icons
|
||||||
|
local textAhead = string_sub(newText, current)
|
||||||
|
local posEnd = Grichelde:CheckForPreversableText(textAhead)
|
||||||
|
if posEnd > 0 then
|
||||||
|
local textBehind = string_sub(newText, lastStart, current - 1)
|
||||||
|
local replacement = Grichelde:ReplaceCharacters(textBehind)
|
||||||
|
local preservedText = string_sub(textAhead, 1, posEnd)
|
||||||
|
|
||||||
|
finalText = finalText .. replacement .. preservedText
|
||||||
|
current = current + posEnd
|
||||||
|
lastStart = current
|
||||||
|
else
|
||||||
|
-- no corresponding end was found to start pattern, continue loop with next char
|
||||||
|
current = current + 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- cleanup to the end
|
||||||
|
local remainingText = newText
|
||||||
|
if lastStart ~= 1 then
|
||||||
|
remainingText = string_sub(newText, lastStart)
|
||||||
|
end
|
||||||
|
|
||||||
|
local replacement = Grichelde:ReplaceCharacters(remainingText)
|
||||||
|
finalText = finalText .. replacement
|
||||||
|
return finalText
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Checks if the text starts with a preversable ignore pattern, such as itemLinks, textures or raid target icons
|
||||||
|
--- @param text string
|
||||||
|
--- @return number
|
||||||
|
function Grichelde:CheckForPreversableText(text)
|
||||||
|
-- Calling find on ever pattern might be inefficient but its way less code.
|
||||||
|
for _, pattern in ipairs(Grichelde_IgnorePatterns) do
|
||||||
|
local pos1, pos2 = string_find(text, pattern)
|
||||||
|
if pos1 == 1 and pos2 ~= nil then
|
||||||
|
return pos2
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return 0
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Replaces all character occurrences for which replacements have been defined in the options
|
||||||
|
--- @param text string
|
||||||
|
--- @return string
|
||||||
|
function Grichelde:ReplaceCharacters(text)
|
||||||
|
-- todo: read from options
|
||||||
|
-- todo: case (in)sensitivity
|
||||||
|
local replacement = text
|
||||||
|
replacement = string_gsub(replacement, "s", "ch")
|
||||||
|
replacement = string_gsub(replacement, "S", "Ch")
|
||||||
|
replacement = string_gsub(replacement, "t", "k")
|
||||||
|
replacement = string_gsub(replacement, "T", "K")
|
||||||
|
return replacement
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
--[[ Interface Options Window ]] --
|
||||||
|
function Grichelde:CreateInterfaceOptions()
|
||||||
|
local cfgFrame = CreateFrame("FRAME", nil, UIParent)
|
||||||
|
cfgFrame.name = "Grichelde"
|
||||||
|
|
||||||
|
local cfgFrameHeader = cfgFrame:CreateFontString("OVERLAY", nil, "GameFontNormalLarge")
|
||||||
|
cfgFrameHeader:SetPoint("TOPLEFT", 15, -15)
|
||||||
|
cfgFrameHeader:SetText(self.Version)
|
||||||
|
|
||||||
|
local cfgAutoSelectDict = CreateFrame("CHECKBUTTON", "Misspelled_cfgAutoSelectDict", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgAutoSelectDict:SetPoint("TOPLEFT", 20, -40)
|
||||||
|
Misspelled_cfgAutoSelectDictText:SetText(L["Auto Select Dictionary to Load"])
|
||||||
|
Misspelled_cfgAutoSelectDict:SetChecked(Misspelled_DB.AutoSelectDictionary)
|
||||||
|
Misspelled_cfgAutoSelectDict:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
Misspelled_DB.AutoSelectDictionary = not Misspelled_DB.AutoSelectDictionary
|
||||||
|
--Toggle the sub options
|
||||||
|
if Misspelled_DB.AutoSelectDictionary == true then
|
||||||
|
Misspelled_cfgDictdeDE:Disable()
|
||||||
|
Misspelled_cfgDictenGB:Disable()
|
||||||
|
Misspelled_cfgDictenUS:Disable()
|
||||||
|
Misspelled_cfgDictesES:Disable()
|
||||||
|
Misspelled_cfgDictfrFR:Disable()
|
||||||
|
Misspelled_cfgDictruRU:Disable()
|
||||||
|
Misspelled_cfgDictitIT:Disable()
|
||||||
|
else
|
||||||
|
Misspelled_cfgDictdeDE:Enable()
|
||||||
|
Misspelled_cfgDictenGB:Enable()
|
||||||
|
Misspelled_cfgDictenUS:Enable()
|
||||||
|
Misspelled_cfgDictesES:Enable()
|
||||||
|
Misspelled_cfgDictfrFR:Enable()
|
||||||
|
Misspelled_cfgDictruRU:Enable()
|
||||||
|
Misspelled_cfgDictitIT:Enable()
|
||||||
|
if Misspelled_DB.LoadDictionary == nil or #Misspelled_DB.LoadDictionary == 0 then
|
||||||
|
Misspelled_DB.LoadDictionary = "enUS"
|
||||||
|
Misspelled_cfgDictenUS:setChecked(true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
local cfgDictdeDE = CreateFrame("CHECKBUTTON", "Misspelled_cfgDictdeDE", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgDictdeDE:SetPoint("TOPLEFT", 40, -64)
|
||||||
|
Misspelled_cfgDictdeDEText:SetText("deDE")
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(Misspelled_DB.LoadDictionary == "deDE")
|
||||||
|
Misspelled_cfgDictdeDE:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
Misspelled_DB.LoadDictionary = "deDE"
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(false)
|
||||||
|
Misspelled_cfgDictesES:SetChecked(false)
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(false)
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(false)
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(false)
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
local cfgDictenGB = CreateFrame("CHECKBUTTON", "Misspelled_cfgDictenGB", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgDictenGB:SetPoint("TOPLEFT", 40, -88)
|
||||||
|
Misspelled_cfgDictenGBText:SetText("enGB")
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(Misspelled_DB.LoadDictionary == "enGB")
|
||||||
|
Misspelled_cfgDictenGB:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
Misspelled_DB.LoadDictionary = "enGB"
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(false)
|
||||||
|
Misspelled_cfgDictesES:SetChecked(false)
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(false)
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(false)
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(false)
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
local cfgDictenUS = CreateFrame("CHECKBUTTON", "Misspelled_cfgDictenUS", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgDictenUS:SetPoint("TOPLEFT", 40, -112)
|
||||||
|
Misspelled_cfgDictenUSText:SetText("enUS")
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(Misspelled_DB.LoadDictionary == "enUS")
|
||||||
|
Misspelled_cfgDictenUS:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
Misspelled_DB.LoadDictionary = "enUS"
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(false)
|
||||||
|
Misspelled_cfgDictesES:SetChecked(false)
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(false)
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(false)
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(false)
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
local cfgDictesES = CreateFrame("CHECKBUTTON", "Misspelled_cfgDictesES", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgDictesES:SetPoint("TOPLEFT", 40, -136)
|
||||||
|
Misspelled_cfgDictesESText:SetText("esES")
|
||||||
|
Misspelled_cfgDictesES:SetChecked(Misspelled_DB.LoadDictionary == "esES")
|
||||||
|
Misspelled_cfgDictesES:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
Misspelled_DB.LoadDictionary = "esES"
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(false)
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(false)
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(false)
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(false)
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
local cfgDictfrFR = CreateFrame("CHECKBUTTON", "Misspelled_cfgDictfrFR", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgDictfrFR:SetPoint("TOPLEFT", 40, -160)
|
||||||
|
Misspelled_cfgDictfrFRText:SetText("frFR")
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(Misspelled_DB.LoadDictionary == "frFR")
|
||||||
|
Misspelled_cfgDictfrFR:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
Misspelled_DB.LoadDictionary = "frFR"
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(false)
|
||||||
|
Misspelled_cfgDictesES:SetChecked(false)
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(false)
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(false)
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
local cfgDictruRU = CreateFrame("CHECKBUTTON", "Misspelled_cfgDictruRU", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgDictruRU:SetPoint("TOPLEFT", 40, -184)
|
||||||
|
Misspelled_cfgDictruRUText:SetText("ruRU")
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(Misspelled_DB.LoadDictionary == "ruRU")
|
||||||
|
Misspelled_cfgDictruRU:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
Misspelled_DB.LoadDictionary = "ruRU"
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(false)
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(false)
|
||||||
|
Misspelled_cfgDictesES:SetChecked(false)
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(false)
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
local cfgDictitIT = CreateFrame("CHECKBUTTON", "Misspelled_cfgDictitIT", cfgFrame, "InterfaceOptionsCheckButtonTemplate")
|
||||||
|
Misspelled_cfgDictitIT:SetPoint("TOPLEFT", 40, -208)
|
||||||
|
Misspelled_cfgDictitITText:SetText("itIT")
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(Misspelled_DB.LoadDictionary == "itIT")
|
||||||
|
Misspelled_cfgDictitIT:SetScript("OnClick", function(self)
|
||||||
|
if self:GetChecked() then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
Misspelled_DB.LoadDictionary = "itIT"
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(false)
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(false)
|
||||||
|
Misspelled_cfgDictesES:SetChecked(false)
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(false)
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
--Edit User Dictionary Button
|
||||||
|
local cfgEditUserDict = CreateFrame("Button", "EdutUserDictButton", cfgFrame, "UIPanelButtonTemplate")
|
||||||
|
cfgEditUserDict:SetPoint("TOPLEFT", 20, -287)
|
||||||
|
cfgEditUserDict:SetText(L["Edit User Dictionary..."])
|
||||||
|
cfgEditUserDict:SetWidth(200)
|
||||||
|
cfgEditUserDict:SetHeight(24)
|
||||||
|
cfgEditUserDict:SetScript("OnClick", function(self)
|
||||||
|
--PlaySound("igMainMenuOptionCheckBoxOn")
|
||||||
|
Misspelled:EditUserDict()
|
||||||
|
end)
|
||||||
|
|
||||||
|
--set options on startup
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(false)
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(false)
|
||||||
|
Misspelled_cfgDictesES:SetChecked(false)
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(false)
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(false)
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(false)
|
||||||
|
|
||||||
|
if Misspelled_DB.LoadDictionary == "deDE" then
|
||||||
|
Misspelled_cfgDictdeDE:SetChecked(true)
|
||||||
|
elseif Misspelled_DB.LoadDictionary == "enGB" then
|
||||||
|
Misspelled_cfgDictenGB:SetChecked(true)
|
||||||
|
elseif Misspelled_DB.LoadDictionary == "enUS" then
|
||||||
|
Misspelled_cfgDictenUS:SetChecked(true)
|
||||||
|
elseif Misspelled_DB.LoadDictionary == "esES" then
|
||||||
|
Misspelled_cfgDictesES:SetChecked(true)
|
||||||
|
elseif Misspelled_DB.LoadDictionary == "frFR" then
|
||||||
|
Misspelled_cfgDictfrFR:SetChecked(true)
|
||||||
|
elseif Misspelled_DB.LoadDictionary == "ruRU" then
|
||||||
|
Misspelled_cfgDictruRU:SetChecked(true)
|
||||||
|
elseif Misspelled_DB.LoadDictionary == "itIT" then
|
||||||
|
Misspelled_cfgDictitIT:SetChecked(true)
|
||||||
|
end
|
||||||
|
|
||||||
|
if Misspelled_DB.AutoSelectDictionary == true then
|
||||||
|
Misspelled_cfgDictdeDE:Disable()
|
||||||
|
Misspelled_cfgDictenGB:Disable()
|
||||||
|
Misspelled_cfgDictenUS:Disable()
|
||||||
|
Misspelled_cfgDictesES:Disable()
|
||||||
|
Misspelled_cfgDictfrFR:Disable()
|
||||||
|
Misspelled_cfgDictruRU:Disable()
|
||||||
|
Misspelled_cfgDictitIT:Disable()
|
||||||
|
else
|
||||||
|
Misspelled_cfgDictdeDE:Enable()
|
||||||
|
Misspelled_cfgDictenGB:Enable()
|
||||||
|
Misspelled_cfgDictenUS:Enable()
|
||||||
|
Misspelled_cfgDictesES:Enable()
|
||||||
|
Misspelled_cfgDictfrFR:Enable()
|
||||||
|
Misspelled_cfgDictruRU:Enable()
|
||||||
|
Misspelled_cfgDictitIT:Enable()
|
||||||
|
end
|
||||||
|
|
||||||
|
InterfaceOptions_AddCategory(cfgFrame)
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
-- split first word of a text line
|
||||||
|
function Grichelde:GetNextWord(line)
|
||||||
|
-- need to add a trailing separator to catch the last value.
|
||||||
|
local wordPattern = "[%s%c]*(%w+)[%s%c]*"
|
||||||
|
local s, e = string_find(line, wordPattern)
|
||||||
|
local word = string_sub(line, s, e)
|
||||||
|
local rest = string_sub(line, e)
|
||||||
|
return word, rest
|
||||||
|
end
|
||||||
|
|
||||||
|
function Grichelde:tprint(t, indent, done)
|
||||||
|
-- in case we run it standalone
|
||||||
|
local Note = Note or print
|
||||||
|
|
||||||
|
-- show strings differently to distinguish them from numbers
|
||||||
|
local function show(val)
|
||||||
|
if type(val) == "string" then
|
||||||
|
return '"' .. val .. '"'
|
||||||
|
else
|
||||||
|
return tostring(val)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- entry point here
|
||||||
|
done = done or {}
|
||||||
|
indent = indent or 0
|
||||||
|
for key, value in pairs(t) do
|
||||||
|
print(string_rep(" ", indent)) -- indent it
|
||||||
|
if type(value) == "table" and not done[value] then
|
||||||
|
done[value] = true
|
||||||
|
Note(show(key), ":");
|
||||||
|
Grichelde:tprint(value, indent + 2, done)
|
||||||
|
else
|
||||||
|
print(show(key), "=")
|
||||||
|
print(show(value))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function Grichelde:print(...)
|
||||||
|
SELECTED_DOCK_FRAME:AddMessage(...)
|
||||||
|
end
|
@ -0,0 +1,21 @@
|
|||||||
|
## Interface: 11304
|
||||||
|
|
||||||
|
## Title: Grichelde
|
||||||
|
## Notes: Replaces characters you type in the chat box
|
||||||
|
## Notes-de: Ersetzt eingegebene Zeichen in der Chat-Zeile
|
||||||
|
## Version: 1.0
|
||||||
|
## Author: Teilzeit-Jedi
|
||||||
|
## eMail: tj@teilzeit-jedi.de
|
||||||
|
|
||||||
|
## X-Build: Classic
|
||||||
|
## X-Curse-Project-ID: 385480
|
||||||
|
## X-Category: Chat/Communication
|
||||||
|
## X-Credits: Teilzeit-Jedi, Nathan Pieper
|
||||||
|
|
||||||
|
## SavedVariables: GrichseldeOptions
|
||||||
|
## SavedVariablesPerCharacter: GrichseldeCharOptions
|
||||||
|
|
||||||
|
libs.xml
|
||||||
|
localisation.xml
|
||||||
|
|
||||||
|
Grichelde.lua
|
@ -0,0 +1,11 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
|
||||||
|
<Script file="libs\LibStub\LibStub.lua"/>
|
||||||
|
<Include file="libs\CallbackHandler-1.0\CallbackHandler-1.0.xml"/>
|
||||||
|
<Include file="libs\AceAddon-3.0\AceAddon-3.0.xml"/>
|
||||||
|
<Include file="libs\AceGUI-3.0\AceGUI-3.0.xml"/>
|
||||||
|
<Include file="libs\AceEvent-3.0\AceEvent-3.0.xml" />
|
||||||
|
<Include file="libs\AceLocale-3.0\AceLocale-3.0.xml" />
|
||||||
|
<Include file="libs\AceHook-3.0\AceHook-3.0.xml"/>
|
||||||
|
</Ui>
|
@ -0,0 +1,643 @@
|
|||||||
|
--- **AceAddon-3.0** provides a template for creating addon objects.
|
||||||
|
-- It'll provide you with a set of callback functions that allow you to simplify the loading
|
||||||
|
-- process of your addon.\\
|
||||||
|
-- Callbacks provided are:\\
|
||||||
|
-- * **OnInitialize**, which is called directly after the addon is fully loaded.
|
||||||
|
-- * **OnEnable** which gets called during the PLAYER_LOGIN event, when most of the data provided by the game is already present.
|
||||||
|
-- * **OnDisable**, which is only called when your addon is manually being disabled.
|
||||||
|
-- @usage
|
||||||
|
-- -- A small (but complete) addon, that doesn't do anything,
|
||||||
|
-- -- but shows usage of the callbacks.
|
||||||
|
-- local MyAddon = LibStub("AceAddon-3.0"):NewAddon("MyAddon")
|
||||||
|
--
|
||||||
|
-- function MyAddon:OnInitialize()
|
||||||
|
-- -- do init tasks here, like loading the Saved Variables,
|
||||||
|
-- -- or setting up slash commands.
|
||||||
|
-- end
|
||||||
|
--
|
||||||
|
-- function MyAddon:OnEnable()
|
||||||
|
-- -- Do more initialization here, that really enables the use of your addon.
|
||||||
|
-- -- Register Events, Hook functions, Create Frames, Get information from
|
||||||
|
-- -- the game that wasn't available in OnInitialize
|
||||||
|
-- end
|
||||||
|
--
|
||||||
|
-- function MyAddon:OnDisable()
|
||||||
|
-- -- Unhook, Unregister Events, Hide frames that you created.
|
||||||
|
-- -- You would probably only use an OnDisable if you want to
|
||||||
|
-- -- build a "standby" mode, or be able to toggle modules on/off.
|
||||||
|
-- end
|
||||||
|
-- @class file
|
||||||
|
-- @name AceAddon-3.0.lua
|
||||||
|
-- @release $Id: AceAddon-3.0.lua 1202 2019-05-15 23:11:22Z nevcairiel $
|
||||||
|
|
||||||
|
local MAJOR, MINOR = "AceAddon-3.0", 12
|
||||||
|
local AceAddon, oldminor = LibStub:NewLibrary(MAJOR, MINOR)
|
||||||
|
|
||||||
|
if not AceAddon then return end -- No Upgrade needed.
|
||||||
|
|
||||||
|
AceAddon.frame = AceAddon.frame or CreateFrame("Frame", "AceAddon30Frame") -- Our very own frame
|
||||||
|
AceAddon.addons = AceAddon.addons or {} -- addons in general
|
||||||
|
AceAddon.statuses = AceAddon.statuses or {} -- statuses of addon.
|
||||||
|
AceAddon.initializequeue = AceAddon.initializequeue or {} -- addons that are new and not initialized
|
||||||
|
AceAddon.enablequeue = AceAddon.enablequeue or {} -- addons that are initialized and waiting to be enabled
|
||||||
|
AceAddon.embeds = AceAddon.embeds or setmetatable({}, {__index = function(tbl, key) tbl[key] = {} return tbl[key] end }) -- contains a list of libraries embedded in an addon
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local tinsert, tconcat, tremove = table.insert, table.concat, table.remove
|
||||||
|
local fmt, tostring = string.format, tostring
|
||||||
|
local select, pairs, next, type, unpack = select, pairs, next, type, unpack
|
||||||
|
local loadstring, assert, error = loadstring, assert, error
|
||||||
|
local setmetatable, getmetatable, rawset, rawget = setmetatable, getmetatable, rawset, rawget
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: LibStub, IsLoggedIn, geterrorhandler
|
||||||
|
|
||||||
|
--[[
|
||||||
|
xpcall safecall implementation
|
||||||
|
]]
|
||||||
|
local xpcall = xpcall
|
||||||
|
|
||||||
|
local function errorhandler(err)
|
||||||
|
return geterrorhandler()(err)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function safecall(func, ...)
|
||||||
|
-- we check to see if the func is passed is actually a function here and don't error when it isn't
|
||||||
|
-- this safecall is used for optional functions like OnInitialize OnEnable etc. When they are not
|
||||||
|
-- present execution should continue without hinderance
|
||||||
|
if type(func) == "function" then
|
||||||
|
return xpcall(func, errorhandler, ...)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- local functions that will be implemented further down
|
||||||
|
local Enable, Disable, EnableModule, DisableModule, Embed, NewModule, GetModule, GetName, SetDefaultModuleState, SetDefaultModuleLibraries, SetEnabledState, SetDefaultModulePrototype
|
||||||
|
|
||||||
|
-- used in the addon metatable
|
||||||
|
local function addontostring( self ) return self.name end
|
||||||
|
|
||||||
|
-- Check if the addon is queued for initialization
|
||||||
|
local function queuedForInitialization(addon)
|
||||||
|
for i = 1, #AceAddon.initializequeue do
|
||||||
|
if AceAddon.initializequeue[i] == addon then
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Create a new AceAddon-3.0 addon.
|
||||||
|
-- Any libraries you specified will be embeded, and the addon will be scheduled for
|
||||||
|
-- its OnInitialize and OnEnable callbacks.
|
||||||
|
-- The final addon object, with all libraries embeded, will be returned.
|
||||||
|
-- @paramsig [object ,]name[, lib, ...]
|
||||||
|
-- @param object Table to use as a base for the addon (optional)
|
||||||
|
-- @param name Name of the addon object to create
|
||||||
|
-- @param lib List of libraries to embed into the addon
|
||||||
|
-- @usage
|
||||||
|
-- -- Create a simple addon object
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("MyAddon", "AceEvent-3.0")
|
||||||
|
--
|
||||||
|
-- -- Create a Addon object based on the table of a frame
|
||||||
|
-- local MyFrame = CreateFrame("Frame")
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon(MyFrame, "MyAddon", "AceEvent-3.0")
|
||||||
|
function AceAddon:NewAddon(objectorname, ...)
|
||||||
|
local object,name
|
||||||
|
local i=1
|
||||||
|
if type(objectorname)=="table" then
|
||||||
|
object=objectorname
|
||||||
|
name=...
|
||||||
|
i=2
|
||||||
|
else
|
||||||
|
name=objectorname
|
||||||
|
end
|
||||||
|
if type(name)~="string" then
|
||||||
|
error(("Usage: NewAddon([object,] name, [lib, lib, lib, ...]): 'name' - string expected got '%s'."):format(type(name)), 2)
|
||||||
|
end
|
||||||
|
if self.addons[name] then
|
||||||
|
error(("Usage: NewAddon([object,] name, [lib, lib, lib, ...]): 'name' - Addon '%s' already exists."):format(name), 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
object = object or {}
|
||||||
|
object.name = name
|
||||||
|
|
||||||
|
local addonmeta = {}
|
||||||
|
local oldmeta = getmetatable(object)
|
||||||
|
if oldmeta then
|
||||||
|
for k, v in pairs(oldmeta) do addonmeta[k] = v end
|
||||||
|
end
|
||||||
|
addonmeta.__tostring = addontostring
|
||||||
|
|
||||||
|
setmetatable( object, addonmeta )
|
||||||
|
self.addons[name] = object
|
||||||
|
object.modules = {}
|
||||||
|
object.orderedModules = {}
|
||||||
|
object.defaultModuleLibraries = {}
|
||||||
|
Embed( object ) -- embed NewModule, GetModule methods
|
||||||
|
self:EmbedLibraries(object, select(i,...))
|
||||||
|
|
||||||
|
-- add to queue of addons to be initialized upon ADDON_LOADED
|
||||||
|
tinsert(self.initializequeue, object)
|
||||||
|
return object
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
--- Get the addon object by its name from the internal AceAddon registry.
|
||||||
|
-- Throws an error if the addon object cannot be found (except if silent is set).
|
||||||
|
-- @param name unique name of the addon object
|
||||||
|
-- @param silent if true, the addon is optional, silently return nil if its not found
|
||||||
|
-- @usage
|
||||||
|
-- -- Get the Addon
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
function AceAddon:GetAddon(name, silent)
|
||||||
|
if not silent and not self.addons[name] then
|
||||||
|
error(("Usage: GetAddon(name): 'name' - Cannot find an AceAddon '%s'."):format(tostring(name)), 2)
|
||||||
|
end
|
||||||
|
return self.addons[name]
|
||||||
|
end
|
||||||
|
|
||||||
|
-- - Embed a list of libraries into the specified addon.
|
||||||
|
-- This function will try to embed all of the listed libraries into the addon
|
||||||
|
-- and error if a single one fails.
|
||||||
|
--
|
||||||
|
-- **Note:** This function is for internal use by :NewAddon/:NewModule
|
||||||
|
-- @paramsig addon, [lib, ...]
|
||||||
|
-- @param addon addon object to embed the libs in
|
||||||
|
-- @param lib List of libraries to embed into the addon
|
||||||
|
function AceAddon:EmbedLibraries(addon, ...)
|
||||||
|
for i=1,select("#", ... ) do
|
||||||
|
local libname = select(i, ...)
|
||||||
|
self:EmbedLibrary(addon, libname, false, 4)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- - Embed a library into the addon object.
|
||||||
|
-- This function will check if the specified library is registered with LibStub
|
||||||
|
-- and if it has a :Embed function to call. It'll error if any of those conditions
|
||||||
|
-- fails.
|
||||||
|
--
|
||||||
|
-- **Note:** This function is for internal use by :EmbedLibraries
|
||||||
|
-- @paramsig addon, libname[, silent[, offset]]
|
||||||
|
-- @param addon addon object to embed the library in
|
||||||
|
-- @param libname name of the library to embed
|
||||||
|
-- @param silent marks an embed to fail silently if the library doesn't exist (optional)
|
||||||
|
-- @param offset will push the error messages back to said offset, defaults to 2 (optional)
|
||||||
|
function AceAddon:EmbedLibrary(addon, libname, silent, offset)
|
||||||
|
local lib = LibStub:GetLibrary(libname, true)
|
||||||
|
if not lib and not silent then
|
||||||
|
error(("Usage: EmbedLibrary(addon, libname, silent, offset): 'libname' - Cannot find a library instance of %q."):format(tostring(libname)), offset or 2)
|
||||||
|
elseif lib and type(lib.Embed) == "function" then
|
||||||
|
lib:Embed(addon)
|
||||||
|
tinsert(self.embeds[addon], libname)
|
||||||
|
return true
|
||||||
|
elseif lib then
|
||||||
|
error(("Usage: EmbedLibrary(addon, libname, silent, offset): 'libname' - Library '%s' is not Embed capable"):format(libname), offset or 2)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Return the specified module from an addon object.
|
||||||
|
-- Throws an error if the addon object cannot be found (except if silent is set)
|
||||||
|
-- @name //addon//:GetModule
|
||||||
|
-- @paramsig name[, silent]
|
||||||
|
-- @param name unique name of the module
|
||||||
|
-- @param silent if true, the module is optional, silently return nil if its not found (optional)
|
||||||
|
-- @usage
|
||||||
|
-- -- Get the Addon
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
-- -- Get the Module
|
||||||
|
-- MyModule = MyAddon:GetModule("MyModule")
|
||||||
|
function GetModule(self, name, silent)
|
||||||
|
if not self.modules[name] and not silent then
|
||||||
|
error(("Usage: GetModule(name, silent): 'name' - Cannot find module '%s'."):format(tostring(name)), 2)
|
||||||
|
end
|
||||||
|
return self.modules[name]
|
||||||
|
end
|
||||||
|
|
||||||
|
local function IsModuleTrue(self) return true end
|
||||||
|
|
||||||
|
--- Create a new module for the addon.
|
||||||
|
-- The new module can have its own embeded libraries and/or use a module prototype to be mixed into the module.\\
|
||||||
|
-- A module has the same functionality as a real addon, it can have modules of its own, and has the same API as
|
||||||
|
-- an addon object.
|
||||||
|
-- @name //addon//:NewModule
|
||||||
|
-- @paramsig name[, prototype|lib[, lib, ...]]
|
||||||
|
-- @param name unique name of the module
|
||||||
|
-- @param prototype object to derive this module from, methods and values from this table will be mixed into the module (optional)
|
||||||
|
-- @param lib List of libraries to embed into the addon
|
||||||
|
-- @usage
|
||||||
|
-- -- Create a module with some embeded libraries
|
||||||
|
-- MyModule = MyAddon:NewModule("MyModule", "AceEvent-3.0", "AceHook-3.0")
|
||||||
|
--
|
||||||
|
-- -- Create a module with a prototype
|
||||||
|
-- local prototype = { OnEnable = function(self) print("OnEnable called!") end }
|
||||||
|
-- MyModule = MyAddon:NewModule("MyModule", prototype, "AceEvent-3.0", "AceHook-3.0")
|
||||||
|
function NewModule(self, name, prototype, ...)
|
||||||
|
if type(name) ~= "string" then error(("Usage: NewModule(name, [prototype, [lib, lib, lib, ...]): 'name' - string expected got '%s'."):format(type(name)), 2) end
|
||||||
|
if type(prototype) ~= "string" and type(prototype) ~= "table" and type(prototype) ~= "nil" then error(("Usage: NewModule(name, [prototype, [lib, lib, lib, ...]): 'prototype' - table (prototype), string (lib) or nil expected got '%s'."):format(type(prototype)), 2) end
|
||||||
|
|
||||||
|
if self.modules[name] then error(("Usage: NewModule(name, [prototype, [lib, lib, lib, ...]): 'name' - Module '%s' already exists."):format(name), 2) end
|
||||||
|
|
||||||
|
-- modules are basically addons. We treat them as such. They will be added to the initializequeue properly as well.
|
||||||
|
-- NewModule can only be called after the parent addon is present thus the modules will be initialized after their parent is.
|
||||||
|
local module = AceAddon:NewAddon(fmt("%s_%s", self.name or tostring(self), name))
|
||||||
|
|
||||||
|
module.IsModule = IsModuleTrue
|
||||||
|
module:SetEnabledState(self.defaultModuleState)
|
||||||
|
module.moduleName = name
|
||||||
|
|
||||||
|
if type(prototype) == "string" then
|
||||||
|
AceAddon:EmbedLibraries(module, prototype, ...)
|
||||||
|
else
|
||||||
|
AceAddon:EmbedLibraries(module, ...)
|
||||||
|
end
|
||||||
|
AceAddon:EmbedLibraries(module, unpack(self.defaultModuleLibraries))
|
||||||
|
|
||||||
|
if not prototype or type(prototype) == "string" then
|
||||||
|
prototype = self.defaultModulePrototype or nil
|
||||||
|
end
|
||||||
|
|
||||||
|
if type(prototype) == "table" then
|
||||||
|
local mt = getmetatable(module)
|
||||||
|
mt.__index = prototype
|
||||||
|
setmetatable(module, mt) -- More of a Base class type feel.
|
||||||
|
end
|
||||||
|
|
||||||
|
safecall(self.OnModuleCreated, self, module) -- Was in Ace2 and I think it could be a cool thing to have handy.
|
||||||
|
self.modules[name] = module
|
||||||
|
tinsert(self.orderedModules, module)
|
||||||
|
|
||||||
|
return module
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Returns the real name of the addon or module, without any prefix.
|
||||||
|
-- @name //addon//:GetName
|
||||||
|
-- @paramsig
|
||||||
|
-- @usage
|
||||||
|
-- print(MyAddon:GetName())
|
||||||
|
-- -- prints "MyAddon"
|
||||||
|
function GetName(self)
|
||||||
|
return self.moduleName or self.name
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Enables the Addon, if possible, return true or false depending on success.
|
||||||
|
-- This internally calls AceAddon:EnableAddon(), thus dispatching a OnEnable callback
|
||||||
|
-- and enabling all modules of the addon (unless explicitly disabled).\\
|
||||||
|
-- :Enable() also sets the internal `enableState` variable to true
|
||||||
|
-- @name //addon//:Enable
|
||||||
|
-- @paramsig
|
||||||
|
-- @usage
|
||||||
|
-- -- Enable MyModule
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
-- MyModule = MyAddon:GetModule("MyModule")
|
||||||
|
-- MyModule:Enable()
|
||||||
|
function Enable(self)
|
||||||
|
self:SetEnabledState(true)
|
||||||
|
|
||||||
|
-- nevcairiel 2013-04-27: don't enable an addon/module if its queued for init still
|
||||||
|
-- it'll be enabled after the init process
|
||||||
|
if not queuedForInitialization(self) then
|
||||||
|
return AceAddon:EnableAddon(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Disables the Addon, if possible, return true or false depending on success.
|
||||||
|
-- This internally calls AceAddon:DisableAddon(), thus dispatching a OnDisable callback
|
||||||
|
-- and disabling all modules of the addon.\\
|
||||||
|
-- :Disable() also sets the internal `enableState` variable to false
|
||||||
|
-- @name //addon//:Disable
|
||||||
|
-- @paramsig
|
||||||
|
-- @usage
|
||||||
|
-- -- Disable MyAddon
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
-- MyAddon:Disable()
|
||||||
|
function Disable(self)
|
||||||
|
self:SetEnabledState(false)
|
||||||
|
return AceAddon:DisableAddon(self)
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Enables the Module, if possible, return true or false depending on success.
|
||||||
|
-- Short-hand function that retrieves the module via `:GetModule` and calls `:Enable` on the module object.
|
||||||
|
-- @name //addon//:EnableModule
|
||||||
|
-- @paramsig name
|
||||||
|
-- @usage
|
||||||
|
-- -- Enable MyModule using :GetModule
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
-- MyModule = MyAddon:GetModule("MyModule")
|
||||||
|
-- MyModule:Enable()
|
||||||
|
--
|
||||||
|
-- -- Enable MyModule using the short-hand
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
-- MyAddon:EnableModule("MyModule")
|
||||||
|
function EnableModule(self, name)
|
||||||
|
local module = self:GetModule( name )
|
||||||
|
return module:Enable()
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Disables the Module, if possible, return true or false depending on success.
|
||||||
|
-- Short-hand function that retrieves the module via `:GetModule` and calls `:Disable` on the module object.
|
||||||
|
-- @name //addon//:DisableModule
|
||||||
|
-- @paramsig name
|
||||||
|
-- @usage
|
||||||
|
-- -- Disable MyModule using :GetModule
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
-- MyModule = MyAddon:GetModule("MyModule")
|
||||||
|
-- MyModule:Disable()
|
||||||
|
--
|
||||||
|
-- -- Disable MyModule using the short-hand
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):GetAddon("MyAddon")
|
||||||
|
-- MyAddon:DisableModule("MyModule")
|
||||||
|
function DisableModule(self, name)
|
||||||
|
local module = self:GetModule( name )
|
||||||
|
return module:Disable()
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Set the default libraries to be mixed into all modules created by this object.
|
||||||
|
-- Note that you can only change the default module libraries before any module is created.
|
||||||
|
-- @name //addon//:SetDefaultModuleLibraries
|
||||||
|
-- @paramsig lib[, lib, ...]
|
||||||
|
-- @param lib List of libraries to embed into the addon
|
||||||
|
-- @usage
|
||||||
|
-- -- Create the addon object
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("MyAddon")
|
||||||
|
-- -- Configure default libraries for modules (all modules need AceEvent-3.0)
|
||||||
|
-- MyAddon:SetDefaultModuleLibraries("AceEvent-3.0")
|
||||||
|
-- -- Create a module
|
||||||
|
-- MyModule = MyAddon:NewModule("MyModule")
|
||||||
|
function SetDefaultModuleLibraries(self, ...)
|
||||||
|
if next(self.modules) then
|
||||||
|
error("Usage: SetDefaultModuleLibraries(...): cannot change the module defaults after a module has been registered.", 2)
|
||||||
|
end
|
||||||
|
self.defaultModuleLibraries = {...}
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Set the default state in which new modules are being created.
|
||||||
|
-- Note that you can only change the default state before any module is created.
|
||||||
|
-- @name //addon//:SetDefaultModuleState
|
||||||
|
-- @paramsig state
|
||||||
|
-- @param state Default state for new modules, true for enabled, false for disabled
|
||||||
|
-- @usage
|
||||||
|
-- -- Create the addon object
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("MyAddon")
|
||||||
|
-- -- Set the default state to "disabled"
|
||||||
|
-- MyAddon:SetDefaultModuleState(false)
|
||||||
|
-- -- Create a module and explicilty enable it
|
||||||
|
-- MyModule = MyAddon:NewModule("MyModule")
|
||||||
|
-- MyModule:Enable()
|
||||||
|
function SetDefaultModuleState(self, state)
|
||||||
|
if next(self.modules) then
|
||||||
|
error("Usage: SetDefaultModuleState(state): cannot change the module defaults after a module has been registered.", 2)
|
||||||
|
end
|
||||||
|
self.defaultModuleState = state
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Set the default prototype to use for new modules on creation.
|
||||||
|
-- Note that you can only change the default prototype before any module is created.
|
||||||
|
-- @name //addon//:SetDefaultModulePrototype
|
||||||
|
-- @paramsig prototype
|
||||||
|
-- @param prototype Default prototype for the new modules (table)
|
||||||
|
-- @usage
|
||||||
|
-- -- Define a prototype
|
||||||
|
-- local prototype = { OnEnable = function(self) print("OnEnable called!") end }
|
||||||
|
-- -- Set the default prototype
|
||||||
|
-- MyAddon:SetDefaultModulePrototype(prototype)
|
||||||
|
-- -- Create a module and explicitly Enable it
|
||||||
|
-- MyModule = MyAddon:NewModule("MyModule")
|
||||||
|
-- MyModule:Enable()
|
||||||
|
-- -- should print "OnEnable called!" now
|
||||||
|
-- @see NewModule
|
||||||
|
function SetDefaultModulePrototype(self, prototype)
|
||||||
|
if next(self.modules) then
|
||||||
|
error("Usage: SetDefaultModulePrototype(prototype): cannot change the module defaults after a module has been registered.", 2)
|
||||||
|
end
|
||||||
|
if type(prototype) ~= "table" then
|
||||||
|
error(("Usage: SetDefaultModulePrototype(prototype): 'prototype' - table expected got '%s'."):format(type(prototype)), 2)
|
||||||
|
end
|
||||||
|
self.defaultModulePrototype = prototype
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Set the state of an addon or module
|
||||||
|
-- This should only be called before any enabling actually happend, e.g. in/before OnInitialize.
|
||||||
|
-- @name //addon//:SetEnabledState
|
||||||
|
-- @paramsig state
|
||||||
|
-- @param state the state of an addon or module (enabled=true, disabled=false)
|
||||||
|
function SetEnabledState(self, state)
|
||||||
|
self.enabledState = state
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
--- Return an iterator of all modules associated to the addon.
|
||||||
|
-- @name //addon//:IterateModules
|
||||||
|
-- @paramsig
|
||||||
|
-- @usage
|
||||||
|
-- -- Enable all modules
|
||||||
|
-- for name, module in MyAddon:IterateModules() do
|
||||||
|
-- module:Enable()
|
||||||
|
-- end
|
||||||
|
local function IterateModules(self) return pairs(self.modules) end
|
||||||
|
|
||||||
|
-- Returns an iterator of all embeds in the addon
|
||||||
|
-- @name //addon//:IterateEmbeds
|
||||||
|
-- @paramsig
|
||||||
|
local function IterateEmbeds(self) return pairs(AceAddon.embeds[self]) end
|
||||||
|
|
||||||
|
--- Query the enabledState of an addon.
|
||||||
|
-- @name //addon//:IsEnabled
|
||||||
|
-- @paramsig
|
||||||
|
-- @usage
|
||||||
|
-- if MyAddon:IsEnabled() then
|
||||||
|
-- MyAddon:Disable()
|
||||||
|
-- end
|
||||||
|
local function IsEnabled(self) return self.enabledState end
|
||||||
|
local mixins = {
|
||||||
|
NewModule = NewModule,
|
||||||
|
GetModule = GetModule,
|
||||||
|
Enable = Enable,
|
||||||
|
Disable = Disable,
|
||||||
|
EnableModule = EnableModule,
|
||||||
|
DisableModule = DisableModule,
|
||||||
|
IsEnabled = IsEnabled,
|
||||||
|
SetDefaultModuleLibraries = SetDefaultModuleLibraries,
|
||||||
|
SetDefaultModuleState = SetDefaultModuleState,
|
||||||
|
SetDefaultModulePrototype = SetDefaultModulePrototype,
|
||||||
|
SetEnabledState = SetEnabledState,
|
||||||
|
IterateModules = IterateModules,
|
||||||
|
IterateEmbeds = IterateEmbeds,
|
||||||
|
GetName = GetName,
|
||||||
|
}
|
||||||
|
local function IsModule(self) return false end
|
||||||
|
local pmixins = {
|
||||||
|
defaultModuleState = true,
|
||||||
|
enabledState = true,
|
||||||
|
IsModule = IsModule,
|
||||||
|
}
|
||||||
|
-- Embed( target )
|
||||||
|
-- target (object) - target object to embed aceaddon in
|
||||||
|
--
|
||||||
|
-- this is a local function specifically since it's meant to be only called internally
|
||||||
|
function Embed(target, skipPMixins)
|
||||||
|
for k, v in pairs(mixins) do
|
||||||
|
target[k] = v
|
||||||
|
end
|
||||||
|
if not skipPMixins then
|
||||||
|
for k, v in pairs(pmixins) do
|
||||||
|
target[k] = target[k] or v
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
-- - Initialize the addon after creation.
|
||||||
|
-- This function is only used internally during the ADDON_LOADED event
|
||||||
|
-- It will call the **OnInitialize** function on the addon object (if present),
|
||||||
|
-- and the **OnEmbedInitialize** function on all embeded libraries.
|
||||||
|
--
|
||||||
|
-- **Note:** Do not call this function manually, unless you're absolutely sure that you know what you are doing.
|
||||||
|
-- @param addon addon object to intialize
|
||||||
|
function AceAddon:InitializeAddon(addon)
|
||||||
|
safecall(addon.OnInitialize, addon)
|
||||||
|
|
||||||
|
local embeds = self.embeds[addon]
|
||||||
|
for i = 1, #embeds do
|
||||||
|
local lib = LibStub:GetLibrary(embeds[i], true)
|
||||||
|
if lib then safecall(lib.OnEmbedInitialize, lib, addon) end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- we don't call InitializeAddon on modules specifically, this is handled
|
||||||
|
-- from the event handler and only done _once_
|
||||||
|
end
|
||||||
|
|
||||||
|
-- - Enable the addon after creation.
|
||||||
|
-- Note: This function is only used internally during the PLAYER_LOGIN event, or during ADDON_LOADED,
|
||||||
|
-- if IsLoggedIn() already returns true at that point, e.g. for LoD Addons.
|
||||||
|
-- It will call the **OnEnable** function on the addon object (if present),
|
||||||
|
-- and the **OnEmbedEnable** function on all embeded libraries.\\
|
||||||
|
-- This function does not toggle the enable state of the addon itself, and will return early if the addon is disabled.
|
||||||
|
--
|
||||||
|
-- **Note:** Do not call this function manually, unless you're absolutely sure that you know what you are doing.
|
||||||
|
-- Use :Enable on the addon itself instead.
|
||||||
|
-- @param addon addon object to enable
|
||||||
|
function AceAddon:EnableAddon(addon)
|
||||||
|
if type(addon) == "string" then addon = AceAddon:GetAddon(addon) end
|
||||||
|
if self.statuses[addon.name] or not addon.enabledState then return false end
|
||||||
|
|
||||||
|
-- set the statuses first, before calling the OnEnable. this allows for Disabling of the addon in OnEnable.
|
||||||
|
self.statuses[addon.name] = true
|
||||||
|
|
||||||
|
safecall(addon.OnEnable, addon)
|
||||||
|
|
||||||
|
-- make sure we're still enabled before continueing
|
||||||
|
if self.statuses[addon.name] then
|
||||||
|
local embeds = self.embeds[addon]
|
||||||
|
for i = 1, #embeds do
|
||||||
|
local lib = LibStub:GetLibrary(embeds[i], true)
|
||||||
|
if lib then safecall(lib.OnEmbedEnable, lib, addon) end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- enable possible modules.
|
||||||
|
local modules = addon.orderedModules
|
||||||
|
for i = 1, #modules do
|
||||||
|
self:EnableAddon(modules[i])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return self.statuses[addon.name] -- return true if we're disabled
|
||||||
|
end
|
||||||
|
|
||||||
|
-- - Disable the addon
|
||||||
|
-- Note: This function is only used internally.
|
||||||
|
-- It will call the **OnDisable** function on the addon object (if present),
|
||||||
|
-- and the **OnEmbedDisable** function on all embeded libraries.\\
|
||||||
|
-- This function does not toggle the enable state of the addon itself, and will return early if the addon is still enabled.
|
||||||
|
--
|
||||||
|
-- **Note:** Do not call this function manually, unless you're absolutely sure that you know what you are doing.
|
||||||
|
-- Use :Disable on the addon itself instead.
|
||||||
|
-- @param addon addon object to enable
|
||||||
|
function AceAddon:DisableAddon(addon)
|
||||||
|
if type(addon) == "string" then addon = AceAddon:GetAddon(addon) end
|
||||||
|
if not self.statuses[addon.name] then return false end
|
||||||
|
|
||||||
|
-- set statuses first before calling OnDisable, this allows for aborting the disable in OnDisable.
|
||||||
|
self.statuses[addon.name] = false
|
||||||
|
|
||||||
|
safecall( addon.OnDisable, addon )
|
||||||
|
|
||||||
|
-- make sure we're still disabling...
|
||||||
|
if not self.statuses[addon.name] then
|
||||||
|
local embeds = self.embeds[addon]
|
||||||
|
for i = 1, #embeds do
|
||||||
|
local lib = LibStub:GetLibrary(embeds[i], true)
|
||||||
|
if lib then safecall(lib.OnEmbedDisable, lib, addon) end
|
||||||
|
end
|
||||||
|
-- disable possible modules.
|
||||||
|
local modules = addon.orderedModules
|
||||||
|
for i = 1, #modules do
|
||||||
|
self:DisableAddon(modules[i])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
return not self.statuses[addon.name] -- return true if we're disabled
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Get an iterator over all registered addons.
|
||||||
|
-- @usage
|
||||||
|
-- -- Print a list of all installed AceAddon's
|
||||||
|
-- for name, addon in AceAddon:IterateAddons() do
|
||||||
|
-- print("Addon: " .. name)
|
||||||
|
-- end
|
||||||
|
function AceAddon:IterateAddons() return pairs(self.addons) end
|
||||||
|
|
||||||
|
--- Get an iterator over the internal status registry.
|
||||||
|
-- @usage
|
||||||
|
-- -- Print a list of all enabled addons
|
||||||
|
-- for name, status in AceAddon:IterateAddonStatus() do
|
||||||
|
-- if status then
|
||||||
|
-- print("EnabledAddon: " .. name)
|
||||||
|
-- end
|
||||||
|
-- end
|
||||||
|
function AceAddon:IterateAddonStatus() return pairs(self.statuses) end
|
||||||
|
|
||||||
|
-- Following Iterators are deprecated, and their addon specific versions should be used
|
||||||
|
-- e.g. addon:IterateEmbeds() instead of :IterateEmbedsOnAddon(addon)
|
||||||
|
function AceAddon:IterateEmbedsOnAddon(addon) return pairs(self.embeds[addon]) end
|
||||||
|
function AceAddon:IterateModulesOfAddon(addon) return pairs(addon.modules) end
|
||||||
|
|
||||||
|
-- Event Handling
|
||||||
|
local function onEvent(this, event, arg1)
|
||||||
|
-- 2011-08-17 nevcairiel - ignore the load event of Blizzard_DebugTools, so a potential startup error isn't swallowed up
|
||||||
|
if (event == "ADDON_LOADED" and arg1 ~= "Blizzard_DebugTools") or event == "PLAYER_LOGIN" then
|
||||||
|
-- if a addon loads another addon, recursion could happen here, so we need to validate the table on every iteration
|
||||||
|
while(#AceAddon.initializequeue > 0) do
|
||||||
|
local addon = tremove(AceAddon.initializequeue, 1)
|
||||||
|
-- this might be an issue with recursion - TODO: validate
|
||||||
|
if event == "ADDON_LOADED" then addon.baseName = arg1 end
|
||||||
|
AceAddon:InitializeAddon(addon)
|
||||||
|
tinsert(AceAddon.enablequeue, addon)
|
||||||
|
end
|
||||||
|
|
||||||
|
if IsLoggedIn() then
|
||||||
|
while(#AceAddon.enablequeue > 0) do
|
||||||
|
local addon = tremove(AceAddon.enablequeue, 1)
|
||||||
|
AceAddon:EnableAddon(addon)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
AceAddon.frame:RegisterEvent("ADDON_LOADED")
|
||||||
|
AceAddon.frame:RegisterEvent("PLAYER_LOGIN")
|
||||||
|
AceAddon.frame:SetScript("OnEvent", onEvent)
|
||||||
|
|
||||||
|
-- upgrade embeded
|
||||||
|
for name, addon in pairs(AceAddon.addons) do
|
||||||
|
Embed(addon, true)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- 2010-10-27 nevcairiel - add new "orderedModules" table
|
||||||
|
if oldminor and oldminor < 10 then
|
||||||
|
for name, addon in pairs(AceAddon.addons) do
|
||||||
|
addon.orderedModules = {}
|
||||||
|
for module_name, module in pairs(addon.modules) do
|
||||||
|
tinsert(addon.orderedModules, module)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
@ -0,0 +1,4 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
<Script file="AceAddon-3.0.lua"/>
|
||||||
|
</Ui>
|
@ -0,0 +1,126 @@
|
|||||||
|
--- AceEvent-3.0 provides event registration and secure dispatching.
|
||||||
|
-- All dispatching is done using **CallbackHandler-1.0**. AceEvent is a simple wrapper around
|
||||||
|
-- CallbackHandler, and dispatches all game events or addon message to the registrees.
|
||||||
|
--
|
||||||
|
-- **AceEvent-3.0** can be embeded into your addon, either explicitly by calling AceEvent:Embed(MyAddon) or by
|
||||||
|
-- specifying it as an embeded library in your AceAddon. All functions will be available on your addon object
|
||||||
|
-- and can be accessed directly, without having to explicitly call AceEvent itself.\\
|
||||||
|
-- It is recommended to embed AceEvent, otherwise you'll have to specify a custom `self` on all calls you
|
||||||
|
-- make into AceEvent.
|
||||||
|
-- @class file
|
||||||
|
-- @name AceEvent-3.0
|
||||||
|
-- @release $Id: AceEvent-3.0.lua 1202 2019-05-15 23:11:22Z nevcairiel $
|
||||||
|
local CallbackHandler = LibStub("CallbackHandler-1.0")
|
||||||
|
|
||||||
|
local MAJOR, MINOR = "AceEvent-3.0", 4
|
||||||
|
local AceEvent = LibStub:NewLibrary(MAJOR, MINOR)
|
||||||
|
|
||||||
|
if not AceEvent then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
AceEvent.frame = AceEvent.frame or CreateFrame("Frame", "AceEvent30Frame") -- our event frame
|
||||||
|
AceEvent.embeds = AceEvent.embeds or {} -- what objects embed this lib
|
||||||
|
|
||||||
|
-- APIs and registry for blizzard events, using CallbackHandler lib
|
||||||
|
if not AceEvent.events then
|
||||||
|
AceEvent.events = CallbackHandler:New(AceEvent,
|
||||||
|
"RegisterEvent", "UnregisterEvent", "UnregisterAllEvents")
|
||||||
|
end
|
||||||
|
|
||||||
|
function AceEvent.events:OnUsed(target, eventname)
|
||||||
|
AceEvent.frame:RegisterEvent(eventname)
|
||||||
|
end
|
||||||
|
|
||||||
|
function AceEvent.events:OnUnused(target, eventname)
|
||||||
|
AceEvent.frame:UnregisterEvent(eventname)
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
-- APIs and registry for IPC messages, using CallbackHandler lib
|
||||||
|
if not AceEvent.messages then
|
||||||
|
AceEvent.messages = CallbackHandler:New(AceEvent,
|
||||||
|
"RegisterMessage", "UnregisterMessage", "UnregisterAllMessages"
|
||||||
|
)
|
||||||
|
AceEvent.SendMessage = AceEvent.messages.Fire
|
||||||
|
end
|
||||||
|
|
||||||
|
--- embedding and embed handling
|
||||||
|
local mixins = {
|
||||||
|
"RegisterEvent", "UnregisterEvent",
|
||||||
|
"RegisterMessage", "UnregisterMessage",
|
||||||
|
"SendMessage",
|
||||||
|
"UnregisterAllEvents", "UnregisterAllMessages",
|
||||||
|
}
|
||||||
|
|
||||||
|
--- Register for a Blizzard Event.
|
||||||
|
-- The callback will be called with the optional `arg` as the first argument (if supplied), and the event name as the second (or first, if no arg was supplied)
|
||||||
|
-- Any arguments to the event will be passed on after that.
|
||||||
|
-- @name AceEvent:RegisterEvent
|
||||||
|
-- @class function
|
||||||
|
-- @paramsig event[, callback [, arg]]
|
||||||
|
-- @param event The event to register for
|
||||||
|
-- @param callback The callback function to call when the event is triggered (funcref or method, defaults to a method with the event name)
|
||||||
|
-- @param arg An optional argument to pass to the callback function
|
||||||
|
|
||||||
|
--- Unregister an event.
|
||||||
|
-- @name AceEvent:UnregisterEvent
|
||||||
|
-- @class function
|
||||||
|
-- @paramsig event
|
||||||
|
-- @param event The event to unregister
|
||||||
|
|
||||||
|
--- Register for a custom AceEvent-internal message.
|
||||||
|
-- The callback will be called with the optional `arg` as the first argument (if supplied), and the event name as the second (or first, if no arg was supplied)
|
||||||
|
-- Any arguments to the event will be passed on after that.
|
||||||
|
-- @name AceEvent:RegisterMessage
|
||||||
|
-- @class function
|
||||||
|
-- @paramsig message[, callback [, arg]]
|
||||||
|
-- @param message The message to register for
|
||||||
|
-- @param callback The callback function to call when the message is triggered (funcref or method, defaults to a method with the event name)
|
||||||
|
-- @param arg An optional argument to pass to the callback function
|
||||||
|
|
||||||
|
--- Unregister a message
|
||||||
|
-- @name AceEvent:UnregisterMessage
|
||||||
|
-- @class function
|
||||||
|
-- @paramsig message
|
||||||
|
-- @param message The message to unregister
|
||||||
|
|
||||||
|
--- Send a message over the AceEvent-3.0 internal message system to other addons registered for this message.
|
||||||
|
-- @name AceEvent:SendMessage
|
||||||
|
-- @class function
|
||||||
|
-- @paramsig message, ...
|
||||||
|
-- @param message The message to send
|
||||||
|
-- @param ... Any arguments to the message
|
||||||
|
|
||||||
|
|
||||||
|
-- Embeds AceEvent into the target object making the functions from the mixins list available on target:..
|
||||||
|
-- @param target target object to embed AceEvent in
|
||||||
|
function AceEvent:Embed(target)
|
||||||
|
for k, v in pairs(mixins) do
|
||||||
|
target[v] = self[v]
|
||||||
|
end
|
||||||
|
self.embeds[target] = true
|
||||||
|
return target
|
||||||
|
end
|
||||||
|
|
||||||
|
-- AceEvent:OnEmbedDisable( target )
|
||||||
|
-- target (object) - target object that is being disabled
|
||||||
|
--
|
||||||
|
-- Unregister all events messages etc when the target disables.
|
||||||
|
-- this method should be called by the target manually or by an addon framework
|
||||||
|
function AceEvent:OnEmbedDisable(target)
|
||||||
|
target:UnregisterAllEvents()
|
||||||
|
target:UnregisterAllMessages()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Script to fire blizzard events into the event listeners
|
||||||
|
local events = AceEvent.events
|
||||||
|
AceEvent.frame:SetScript("OnEvent", function(this, event, ...)
|
||||||
|
events:Fire(event, ...)
|
||||||
|
end)
|
||||||
|
|
||||||
|
--- Finally: upgrade our old embeds
|
||||||
|
for target, v in pairs(AceEvent.embeds) do
|
||||||
|
AceEvent:Embed(target)
|
||||||
|
end
|
@ -0,0 +1,4 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
<Script file="AceEvent-3.0.lua"/>
|
||||||
|
</Ui>
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,28 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
<Script file="AceGUI-3.0.lua"/>
|
||||||
|
<!-- Container -->
|
||||||
|
<Script file="widgets\AceGUIContainer-BlizOptionsGroup.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-DropDownGroup.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-Frame.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-InlineGroup.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-ScrollFrame.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-SimpleGroup.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-TabGroup.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-TreeGroup.lua"/>
|
||||||
|
<Script file="widgets\AceGUIContainer-Window.lua"/>
|
||||||
|
<!-- Widgets -->
|
||||||
|
<Script file="widgets\AceGUIWidget-Button.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-CheckBox.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-ColorPicker.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-DropDown.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-DropDown-Items.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-EditBox.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-Heading.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-Icon.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-InteractiveLabel.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-Keybinding.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-Label.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-MultiLineEditBox.lua"/>
|
||||||
|
<Script file="widgets\AceGUIWidget-Slider.lua"/>
|
||||||
|
</Ui>
|
@ -0,0 +1,138 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
BlizOptionsGroup Container
|
||||||
|
Simple container widget for the integration of AceGUI into the Blizzard Interface Options
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "BlizOptionsGroup", 21
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame = CreateFrame
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
|
||||||
|
local function OnShow(frame)
|
||||||
|
frame.obj:Fire("OnShow")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnHide(frame)
|
||||||
|
frame.obj:Fire("OnHide")
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
|
||||||
|
local function okay(frame)
|
||||||
|
frame.obj:Fire("okay")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function cancel(frame)
|
||||||
|
frame.obj:Fire("cancel")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function default(frame)
|
||||||
|
frame.obj:Fire("default")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function refresh(frame)
|
||||||
|
frame.obj:Fire("refresh")
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetName()
|
||||||
|
self:SetTitle()
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
local contentwidth = width - 63
|
||||||
|
if contentwidth < 0 then
|
||||||
|
contentwidth = 0
|
||||||
|
end
|
||||||
|
content:SetWidth(contentwidth)
|
||||||
|
content.width = contentwidth
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
local contentheight = height - 26
|
||||||
|
if contentheight < 0 then
|
||||||
|
contentheight = 0
|
||||||
|
end
|
||||||
|
content:SetHeight(contentheight)
|
||||||
|
content.height = contentheight
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetName"] = function(self, name, parent)
|
||||||
|
self.frame.name = name
|
||||||
|
self.frame.parent = parent
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetTitle"] = function(self, title)
|
||||||
|
local content = self.content
|
||||||
|
content:ClearAllPoints()
|
||||||
|
if not title or title == "" then
|
||||||
|
content:SetPoint("TOPLEFT", 10, -10)
|
||||||
|
self.label:SetText("")
|
||||||
|
else
|
||||||
|
content:SetPoint("TOPLEFT", 10, -40)
|
||||||
|
self.label:SetText(title)
|
||||||
|
end
|
||||||
|
content:SetPoint("BOTTOMRIGHT", -10, 10)
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame")
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
-- support functions for the Blizzard Interface Options
|
||||||
|
frame.okay = okay
|
||||||
|
frame.cancel = cancel
|
||||||
|
frame.default = default
|
||||||
|
frame.refresh = refresh
|
||||||
|
|
||||||
|
frame:SetScript("OnHide", OnHide)
|
||||||
|
frame:SetScript("OnShow", OnShow)
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "OVERLAY", "GameFontNormalLarge")
|
||||||
|
label:SetPoint("TOPLEFT", 10, -15)
|
||||||
|
label:SetPoint("BOTTOMRIGHT", frame, "TOPRIGHT", 10, -45)
|
||||||
|
label:SetJustifyH("LEFT")
|
||||||
|
label:SetJustifyV("TOP")
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame", nil, frame)
|
||||||
|
content:SetPoint("TOPLEFT", 10, -10)
|
||||||
|
content:SetPoint("BOTTOMRIGHT", -10, 10)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
label = label,
|
||||||
|
frame = frame,
|
||||||
|
content = content,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,157 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
DropdownGroup Container
|
||||||
|
Container controlled by a dropdown on the top.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "DropdownGroup", 21
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local assert, pairs, type = assert, pairs, type
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame = CreateFrame
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function SelectedGroup(self, event, value)
|
||||||
|
local group = self.parentgroup
|
||||||
|
local status = group.status or group.localstatus
|
||||||
|
status.selected = value
|
||||||
|
self.parentgroup:Fire("OnGroupSelected", value)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self.dropdown:SetText("")
|
||||||
|
self:SetDropdownWidth(200)
|
||||||
|
self:SetTitle("")
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnRelease"] = function(self)
|
||||||
|
self.dropdown.list = nil
|
||||||
|
self.status = nil
|
||||||
|
for k in pairs(self.localstatus) do
|
||||||
|
self.localstatus[k] = nil
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetTitle"] = function(self, title)
|
||||||
|
self.titletext:SetText(title)
|
||||||
|
self.dropdown.frame:ClearAllPoints()
|
||||||
|
if title and title ~= "" then
|
||||||
|
self.dropdown.frame:SetPoint("TOPRIGHT", -2, 0)
|
||||||
|
else
|
||||||
|
self.dropdown.frame:SetPoint("TOPLEFT", -1, 0)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetGroupList"] = function(self,list,order)
|
||||||
|
self.dropdown:SetList(list,order)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetStatusTable"] = function(self, status)
|
||||||
|
assert(type(status) == "table")
|
||||||
|
self.status = status
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetGroup"] = function(self,group)
|
||||||
|
self.dropdown:SetValue(group)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
status.selected = group
|
||||||
|
self:Fire("OnGroupSelected", group)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
local contentwidth = width - 26
|
||||||
|
if contentwidth < 0 then
|
||||||
|
contentwidth = 0
|
||||||
|
end
|
||||||
|
content:SetWidth(contentwidth)
|
||||||
|
content.width = contentwidth
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
local contentheight = height - 63
|
||||||
|
if contentheight < 0 then
|
||||||
|
contentheight = 0
|
||||||
|
end
|
||||||
|
content:SetHeight(contentheight)
|
||||||
|
content.height = contentheight
|
||||||
|
end,
|
||||||
|
|
||||||
|
["LayoutFinished"] = function(self, width, height)
|
||||||
|
self:SetHeight((height or 0) + 63)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDropdownWidth"] = function(self, width)
|
||||||
|
self.dropdown:SetWidth(width)
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local PaneBackdrop = {
|
||||||
|
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
edgeFile = "Interface\\Tooltips\\UI-Tooltip-Border",
|
||||||
|
tile = true, tileSize = 16, edgeSize = 16,
|
||||||
|
insets = { left = 3, right = 3, top = 5, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame")
|
||||||
|
frame:SetHeight(100)
|
||||||
|
frame:SetWidth(100)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
|
||||||
|
local titletext = frame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||||
|
titletext:SetPoint("TOPLEFT", 4, -5)
|
||||||
|
titletext:SetPoint("TOPRIGHT", -4, -5)
|
||||||
|
titletext:SetJustifyH("LEFT")
|
||||||
|
titletext:SetHeight(18)
|
||||||
|
|
||||||
|
local dropdown = AceGUI:Create("Dropdown")
|
||||||
|
dropdown.frame:SetParent(frame)
|
||||||
|
dropdown.frame:SetFrameLevel(dropdown.frame:GetFrameLevel() + 2)
|
||||||
|
dropdown:SetCallback("OnValueChanged", SelectedGroup)
|
||||||
|
dropdown.frame:SetPoint("TOPLEFT", -1, 0)
|
||||||
|
dropdown.frame:Show()
|
||||||
|
dropdown:SetLabel("")
|
||||||
|
|
||||||
|
local border = CreateFrame("Frame", nil, frame)
|
||||||
|
border:SetPoint("TOPLEFT", 0, -26)
|
||||||
|
border:SetPoint("BOTTOMRIGHT", 0, 3)
|
||||||
|
border:SetBackdrop(PaneBackdrop)
|
||||||
|
border:SetBackdropColor(0.1,0.1,0.1,0.5)
|
||||||
|
border:SetBackdropBorderColor(0.4,0.4,0.4)
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame", nil, border)
|
||||||
|
content:SetPoint("TOPLEFT", 10, -10)
|
||||||
|
content:SetPoint("BOTTOMRIGHT", -10, 10)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
frame = frame,
|
||||||
|
localstatus = {},
|
||||||
|
titletext = titletext,
|
||||||
|
dropdown = dropdown,
|
||||||
|
border = border,
|
||||||
|
content = content,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
dropdown.parentgroup = widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,316 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Frame Container
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "Frame", 26
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs, assert, type = pairs, assert, type
|
||||||
|
local wipe = table.wipe
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: CLOSE
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Button_OnClick(frame)
|
||||||
|
PlaySound(799) -- SOUNDKIT.GS_TITLE_OPTION_EXIT
|
||||||
|
frame.obj:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Frame_OnShow(frame)
|
||||||
|
frame.obj:Fire("OnShow")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Frame_OnClose(frame)
|
||||||
|
frame.obj:Fire("OnClose")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Frame_OnMouseDown(frame)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Title_OnMouseDown(frame)
|
||||||
|
frame:GetParent():StartMoving()
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function MoverSizer_OnMouseUp(mover)
|
||||||
|
local frame = mover:GetParent()
|
||||||
|
frame:StopMovingOrSizing()
|
||||||
|
local self = frame.obj
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
status.width = frame:GetWidth()
|
||||||
|
status.height = frame:GetHeight()
|
||||||
|
status.top = frame:GetTop()
|
||||||
|
status.left = frame:GetLeft()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function SizerSE_OnMouseDown(frame)
|
||||||
|
frame:GetParent():StartSizing("BOTTOMRIGHT")
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function SizerS_OnMouseDown(frame)
|
||||||
|
frame:GetParent():StartSizing("BOTTOM")
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function SizerE_OnMouseDown(frame)
|
||||||
|
frame:GetParent():StartSizing("RIGHT")
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function StatusBar_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnterStatusBar")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function StatusBar_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeaveStatusBar")
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self.frame:SetParent(UIParent)
|
||||||
|
self.frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
self:SetTitle()
|
||||||
|
self:SetStatusText()
|
||||||
|
self:ApplyStatus()
|
||||||
|
self:Show()
|
||||||
|
self:EnableResize(true)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnRelease"] = function(self)
|
||||||
|
self.status = nil
|
||||||
|
wipe(self.localstatus)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
local contentwidth = width - 34
|
||||||
|
if contentwidth < 0 then
|
||||||
|
contentwidth = 0
|
||||||
|
end
|
||||||
|
content:SetWidth(contentwidth)
|
||||||
|
content.width = contentwidth
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
local contentheight = height - 57
|
||||||
|
if contentheight < 0 then
|
||||||
|
contentheight = 0
|
||||||
|
end
|
||||||
|
content:SetHeight(contentheight)
|
||||||
|
content.height = contentheight
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetTitle"] = function(self, title)
|
||||||
|
self.titletext:SetText(title)
|
||||||
|
self.titlebg:SetWidth((self.titletext:GetWidth() or 0) + 10)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetStatusText"] = function(self, text)
|
||||||
|
self.statustext:SetText(text)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["Hide"] = function(self)
|
||||||
|
self.frame:Hide()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["Show"] = function(self)
|
||||||
|
self.frame:Show()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["EnableResize"] = function(self, state)
|
||||||
|
local func = state and "Show" or "Hide"
|
||||||
|
self.sizer_se[func](self.sizer_se)
|
||||||
|
self.sizer_s[func](self.sizer_s)
|
||||||
|
self.sizer_e[func](self.sizer_e)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- called to set an external table to store status in
|
||||||
|
["SetStatusTable"] = function(self, status)
|
||||||
|
assert(type(status) == "table")
|
||||||
|
self.status = status
|
||||||
|
self:ApplyStatus()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["ApplyStatus"] = function(self)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local frame = self.frame
|
||||||
|
self:SetWidth(status.width or 700)
|
||||||
|
self:SetHeight(status.height or 500)
|
||||||
|
frame:ClearAllPoints()
|
||||||
|
if status.top and status.left then
|
||||||
|
frame:SetPoint("TOP", UIParent, "BOTTOM", 0, status.top)
|
||||||
|
frame:SetPoint("LEFT", UIParent, "LEFT", status.left, 0)
|
||||||
|
else
|
||||||
|
frame:SetPoint("CENTER")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local FrameBackdrop = {
|
||||||
|
bgFile = "Interface\\DialogFrame\\UI-DialogBox-Background",
|
||||||
|
edgeFile = "Interface\\DialogFrame\\UI-DialogBox-Border",
|
||||||
|
tile = true, tileSize = 32, edgeSize = 32,
|
||||||
|
insets = { left = 8, right = 8, top = 8, bottom = 8 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local PaneBackdrop = {
|
||||||
|
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
edgeFile = "Interface\\Tooltips\\UI-Tooltip-Border",
|
||||||
|
tile = true, tileSize = 16, edgeSize = 16,
|
||||||
|
insets = { left = 3, right = 3, top = 5, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
frame:EnableMouse(true)
|
||||||
|
frame:SetMovable(true)
|
||||||
|
frame:SetResizable(true)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
frame:SetBackdrop(FrameBackdrop)
|
||||||
|
frame:SetBackdropColor(0, 0, 0, 1)
|
||||||
|
frame:SetMinResize(400, 200)
|
||||||
|
frame:SetToplevel(true)
|
||||||
|
frame:SetScript("OnShow", Frame_OnShow)
|
||||||
|
frame:SetScript("OnHide", Frame_OnClose)
|
||||||
|
frame:SetScript("OnMouseDown", Frame_OnMouseDown)
|
||||||
|
|
||||||
|
local closebutton = CreateFrame("Button", nil, frame, "UIPanelButtonTemplate")
|
||||||
|
closebutton:SetScript("OnClick", Button_OnClick)
|
||||||
|
closebutton:SetPoint("BOTTOMRIGHT", -27, 17)
|
||||||
|
closebutton:SetHeight(20)
|
||||||
|
closebutton:SetWidth(100)
|
||||||
|
closebutton:SetText(CLOSE)
|
||||||
|
|
||||||
|
local statusbg = CreateFrame("Button", nil, frame)
|
||||||
|
statusbg:SetPoint("BOTTOMLEFT", 15, 15)
|
||||||
|
statusbg:SetPoint("BOTTOMRIGHT", -132, 15)
|
||||||
|
statusbg:SetHeight(24)
|
||||||
|
statusbg:SetBackdrop(PaneBackdrop)
|
||||||
|
statusbg:SetBackdropColor(0.1,0.1,0.1)
|
||||||
|
statusbg:SetBackdropBorderColor(0.4,0.4,0.4)
|
||||||
|
statusbg:SetScript("OnEnter", StatusBar_OnEnter)
|
||||||
|
statusbg:SetScript("OnLeave", StatusBar_OnLeave)
|
||||||
|
|
||||||
|
local statustext = statusbg:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||||
|
statustext:SetPoint("TOPLEFT", 7, -2)
|
||||||
|
statustext:SetPoint("BOTTOMRIGHT", -7, 2)
|
||||||
|
statustext:SetHeight(20)
|
||||||
|
statustext:SetJustifyH("LEFT")
|
||||||
|
statustext:SetText("")
|
||||||
|
|
||||||
|
local titlebg = frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
titlebg:SetTexture(131080) -- Interface\\DialogFrame\\UI-DialogBox-Header
|
||||||
|
titlebg:SetTexCoord(0.31, 0.67, 0, 0.63)
|
||||||
|
titlebg:SetPoint("TOP", 0, 12)
|
||||||
|
titlebg:SetWidth(100)
|
||||||
|
titlebg:SetHeight(40)
|
||||||
|
|
||||||
|
local title = CreateFrame("Frame", nil, frame)
|
||||||
|
title:EnableMouse(true)
|
||||||
|
title:SetScript("OnMouseDown", Title_OnMouseDown)
|
||||||
|
title:SetScript("OnMouseUp", MoverSizer_OnMouseUp)
|
||||||
|
title:SetAllPoints(titlebg)
|
||||||
|
|
||||||
|
local titletext = title:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||||
|
titletext:SetPoint("TOP", titlebg, "TOP", 0, -14)
|
||||||
|
|
||||||
|
local titlebg_l = frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
titlebg_l:SetTexture(131080) -- Interface\\DialogFrame\\UI-DialogBox-Header
|
||||||
|
titlebg_l:SetTexCoord(0.21, 0.31, 0, 0.63)
|
||||||
|
titlebg_l:SetPoint("RIGHT", titlebg, "LEFT")
|
||||||
|
titlebg_l:SetWidth(30)
|
||||||
|
titlebg_l:SetHeight(40)
|
||||||
|
|
||||||
|
local titlebg_r = frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
titlebg_r:SetTexture(131080) -- Interface\\DialogFrame\\UI-DialogBox-Header
|
||||||
|
titlebg_r:SetTexCoord(0.67, 0.77, 0, 0.63)
|
||||||
|
titlebg_r:SetPoint("LEFT", titlebg, "RIGHT")
|
||||||
|
titlebg_r:SetWidth(30)
|
||||||
|
titlebg_r:SetHeight(40)
|
||||||
|
|
||||||
|
local sizer_se = CreateFrame("Frame", nil, frame)
|
||||||
|
sizer_se:SetPoint("BOTTOMRIGHT")
|
||||||
|
sizer_se:SetWidth(25)
|
||||||
|
sizer_se:SetHeight(25)
|
||||||
|
sizer_se:EnableMouse()
|
||||||
|
sizer_se:SetScript("OnMouseDown",SizerSE_OnMouseDown)
|
||||||
|
sizer_se:SetScript("OnMouseUp", MoverSizer_OnMouseUp)
|
||||||
|
|
||||||
|
local line1 = sizer_se:CreateTexture(nil, "BACKGROUND")
|
||||||
|
line1:SetWidth(14)
|
||||||
|
line1:SetHeight(14)
|
||||||
|
line1:SetPoint("BOTTOMRIGHT", -8, 8)
|
||||||
|
line1:SetTexture(137057) -- Interface\\Tooltips\\UI-Tooltip-Border
|
||||||
|
local x = 0.1 * 14/17
|
||||||
|
line1:SetTexCoord(0.05 - x, 0.5, 0.05, 0.5 + x, 0.05, 0.5 - x, 0.5 + x, 0.5)
|
||||||
|
|
||||||
|
local line2 = sizer_se:CreateTexture(nil, "BACKGROUND")
|
||||||
|
line2:SetWidth(8)
|
||||||
|
line2:SetHeight(8)
|
||||||
|
line2:SetPoint("BOTTOMRIGHT", -8, 8)
|
||||||
|
line2:SetTexture(137057) -- Interface\\Tooltips\\UI-Tooltip-Border
|
||||||
|
local x = 0.1 * 8/17
|
||||||
|
line2:SetTexCoord(0.05 - x, 0.5, 0.05, 0.5 + x, 0.05, 0.5 - x, 0.5 + x, 0.5)
|
||||||
|
|
||||||
|
local sizer_s = CreateFrame("Frame", nil, frame)
|
||||||
|
sizer_s:SetPoint("BOTTOMRIGHT", -25, 0)
|
||||||
|
sizer_s:SetPoint("BOTTOMLEFT")
|
||||||
|
sizer_s:SetHeight(25)
|
||||||
|
sizer_s:EnableMouse(true)
|
||||||
|
sizer_s:SetScript("OnMouseDown", SizerS_OnMouseDown)
|
||||||
|
sizer_s:SetScript("OnMouseUp", MoverSizer_OnMouseUp)
|
||||||
|
|
||||||
|
local sizer_e = CreateFrame("Frame", nil, frame)
|
||||||
|
sizer_e:SetPoint("BOTTOMRIGHT", 0, 25)
|
||||||
|
sizer_e:SetPoint("TOPRIGHT")
|
||||||
|
sizer_e:SetWidth(25)
|
||||||
|
sizer_e:EnableMouse(true)
|
||||||
|
sizer_e:SetScript("OnMouseDown", SizerE_OnMouseDown)
|
||||||
|
sizer_e:SetScript("OnMouseUp", MoverSizer_OnMouseUp)
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame", nil, frame)
|
||||||
|
content:SetPoint("TOPLEFT", 17, -27)
|
||||||
|
content:SetPoint("BOTTOMRIGHT", -17, 40)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
localstatus = {},
|
||||||
|
titletext = titletext,
|
||||||
|
statustext = statustext,
|
||||||
|
titlebg = titlebg,
|
||||||
|
sizer_se = sizer_se,
|
||||||
|
sizer_s = sizer_s,
|
||||||
|
sizer_e = sizer_e,
|
||||||
|
content = content,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
closebutton.obj, statusbg.obj = widget, widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,103 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
InlineGroup Container
|
||||||
|
Simple container widget that creates a visible "box" with an optional title.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "InlineGroup", 21
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetWidth(300)
|
||||||
|
self:SetHeight(100)
|
||||||
|
self:SetTitle("")
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetTitle"] = function(self,title)
|
||||||
|
self.titletext:SetText(title)
|
||||||
|
end,
|
||||||
|
|
||||||
|
|
||||||
|
["LayoutFinished"] = function(self, width, height)
|
||||||
|
if self.noAutoHeight then return end
|
||||||
|
self:SetHeight((height or 0) + 40)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
local contentwidth = width - 20
|
||||||
|
if contentwidth < 0 then
|
||||||
|
contentwidth = 0
|
||||||
|
end
|
||||||
|
content:SetWidth(contentwidth)
|
||||||
|
content.width = contentwidth
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
local contentheight = height - 20
|
||||||
|
if contentheight < 0 then
|
||||||
|
contentheight = 0
|
||||||
|
end
|
||||||
|
content:SetHeight(contentheight)
|
||||||
|
content.height = contentheight
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local PaneBackdrop = {
|
||||||
|
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
edgeFile = "Interface\\Tooltips\\UI-Tooltip-Border",
|
||||||
|
tile = true, tileSize = 16, edgeSize = 16,
|
||||||
|
insets = { left = 3, right = 3, top = 5, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
|
||||||
|
local titletext = frame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||||
|
titletext:SetPoint("TOPLEFT", 14, 0)
|
||||||
|
titletext:SetPoint("TOPRIGHT", -14, 0)
|
||||||
|
titletext:SetJustifyH("LEFT")
|
||||||
|
titletext:SetHeight(18)
|
||||||
|
|
||||||
|
local border = CreateFrame("Frame", nil, frame)
|
||||||
|
border:SetPoint("TOPLEFT", 0, -17)
|
||||||
|
border:SetPoint("BOTTOMRIGHT", -1, 3)
|
||||||
|
border:SetBackdrop(PaneBackdrop)
|
||||||
|
border:SetBackdropColor(0.1, 0.1, 0.1, 0.5)
|
||||||
|
border:SetBackdropBorderColor(0.4, 0.4, 0.4)
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame", nil, border)
|
||||||
|
content:SetPoint("TOPLEFT", 10, -10)
|
||||||
|
content:SetPoint("BOTTOMRIGHT", -10, 10)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
frame = frame,
|
||||||
|
content = content,
|
||||||
|
titletext = titletext,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,215 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
ScrollFrame Container
|
||||||
|
Plain container that scrolls its content and doesn't grow in height.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "ScrollFrame", 26
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs, assert, type = pairs, assert, type
|
||||||
|
local min, max, floor = math.min, math.max, math.floor
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function FixScrollOnUpdate(frame)
|
||||||
|
frame:SetScript("OnUpdate", nil)
|
||||||
|
frame.obj:FixScroll()
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function ScrollFrame_OnMouseWheel(frame, value)
|
||||||
|
frame.obj:MoveScroll(value)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ScrollFrame_OnSizeChanged(frame)
|
||||||
|
frame:SetScript("OnUpdate", FixScrollOnUpdate)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ScrollBar_OnScrollValueChanged(frame, value)
|
||||||
|
frame.obj:SetScroll(value)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetScroll(0)
|
||||||
|
self.scrollframe:SetScript("OnUpdate", FixScrollOnUpdate)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnRelease"] = function(self)
|
||||||
|
self.status = nil
|
||||||
|
for k in pairs(self.localstatus) do
|
||||||
|
self.localstatus[k] = nil
|
||||||
|
end
|
||||||
|
self.scrollframe:SetPoint("BOTTOMRIGHT")
|
||||||
|
self.scrollbar:Hide()
|
||||||
|
self.scrollBarShown = nil
|
||||||
|
self.content.height, self.content.width, self.content.original_width = nil, nil, nil
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetScroll"] = function(self, value)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local viewheight = self.scrollframe:GetHeight()
|
||||||
|
local height = self.content:GetHeight()
|
||||||
|
local offset
|
||||||
|
|
||||||
|
if viewheight > height then
|
||||||
|
offset = 0
|
||||||
|
else
|
||||||
|
offset = floor((height - viewheight) / 1000.0 * value)
|
||||||
|
end
|
||||||
|
self.content:ClearAllPoints()
|
||||||
|
self.content:SetPoint("TOPLEFT", 0, offset)
|
||||||
|
self.content:SetPoint("TOPRIGHT", 0, offset)
|
||||||
|
status.offset = offset
|
||||||
|
status.scrollvalue = value
|
||||||
|
end,
|
||||||
|
|
||||||
|
["MoveScroll"] = function(self, value)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local height, viewheight = self.scrollframe:GetHeight(), self.content:GetHeight()
|
||||||
|
|
||||||
|
if self.scrollBarShown then
|
||||||
|
local diff = height - viewheight
|
||||||
|
local delta = 1
|
||||||
|
if value < 0 then
|
||||||
|
delta = -1
|
||||||
|
end
|
||||||
|
self.scrollbar:SetValue(min(max(status.scrollvalue + delta*(1000/(diff/45)),0), 1000))
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["FixScroll"] = function(self)
|
||||||
|
if self.updateLock then return end
|
||||||
|
self.updateLock = true
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local height, viewheight = self.scrollframe:GetHeight(), self.content:GetHeight()
|
||||||
|
local offset = status.offset or 0
|
||||||
|
-- Give us a margin of error of 2 pixels to stop some conditions that i would blame on floating point inaccuracys
|
||||||
|
-- No-one is going to miss 2 pixels at the bottom of the frame, anyhow!
|
||||||
|
if viewheight < height + 2 then
|
||||||
|
if self.scrollBarShown then
|
||||||
|
self.scrollBarShown = nil
|
||||||
|
self.scrollbar:Hide()
|
||||||
|
self.scrollbar:SetValue(0)
|
||||||
|
self.scrollframe:SetPoint("BOTTOMRIGHT")
|
||||||
|
if self.content.original_width then
|
||||||
|
self.content.width = self.content.original_width
|
||||||
|
end
|
||||||
|
self:DoLayout()
|
||||||
|
end
|
||||||
|
else
|
||||||
|
if not self.scrollBarShown then
|
||||||
|
self.scrollBarShown = true
|
||||||
|
self.scrollbar:Show()
|
||||||
|
self.scrollframe:SetPoint("BOTTOMRIGHT", -20, 0)
|
||||||
|
if self.content.original_width then
|
||||||
|
self.content.width = self.content.original_width - 20
|
||||||
|
end
|
||||||
|
self:DoLayout()
|
||||||
|
end
|
||||||
|
local value = (offset / (viewheight - height) * 1000)
|
||||||
|
if value > 1000 then value = 1000 end
|
||||||
|
self.scrollbar:SetValue(value)
|
||||||
|
self:SetScroll(value)
|
||||||
|
if value < 1000 then
|
||||||
|
self.content:ClearAllPoints()
|
||||||
|
self.content:SetPoint("TOPLEFT", 0, offset)
|
||||||
|
self.content:SetPoint("TOPRIGHT", 0, offset)
|
||||||
|
status.offset = offset
|
||||||
|
end
|
||||||
|
end
|
||||||
|
self.updateLock = nil
|
||||||
|
end,
|
||||||
|
|
||||||
|
["LayoutFinished"] = function(self, width, height)
|
||||||
|
self.content:SetHeight(height or 0 + 20)
|
||||||
|
|
||||||
|
-- update the scrollframe
|
||||||
|
self:FixScroll()
|
||||||
|
|
||||||
|
-- schedule another update when everything has "settled"
|
||||||
|
self.scrollframe:SetScript("OnUpdate", FixScrollOnUpdate)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetStatusTable"] = function(self, status)
|
||||||
|
assert(type(status) == "table")
|
||||||
|
self.status = status
|
||||||
|
if not status.scrollvalue then
|
||||||
|
status.scrollvalue = 0
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
content.width = width - (self.scrollBarShown and 20 or 0)
|
||||||
|
content.original_width = width
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
content.height = height
|
||||||
|
end
|
||||||
|
}
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
local num = AceGUI:GetNextWidgetNum(Type)
|
||||||
|
|
||||||
|
local scrollframe = CreateFrame("ScrollFrame", nil, frame)
|
||||||
|
scrollframe:SetPoint("TOPLEFT")
|
||||||
|
scrollframe:SetPoint("BOTTOMRIGHT")
|
||||||
|
scrollframe:EnableMouseWheel(true)
|
||||||
|
scrollframe:SetScript("OnMouseWheel", ScrollFrame_OnMouseWheel)
|
||||||
|
scrollframe:SetScript("OnSizeChanged", ScrollFrame_OnSizeChanged)
|
||||||
|
|
||||||
|
local scrollbar = CreateFrame("Slider", ("AceConfigDialogScrollFrame%dScrollBar"):format(num), scrollframe, "UIPanelScrollBarTemplate")
|
||||||
|
scrollbar:SetPoint("TOPLEFT", scrollframe, "TOPRIGHT", 4, -16)
|
||||||
|
scrollbar:SetPoint("BOTTOMLEFT", scrollframe, "BOTTOMRIGHT", 4, 16)
|
||||||
|
scrollbar:SetMinMaxValues(0, 1000)
|
||||||
|
scrollbar:SetValueStep(1)
|
||||||
|
scrollbar:SetValue(0)
|
||||||
|
scrollbar:SetWidth(16)
|
||||||
|
scrollbar:Hide()
|
||||||
|
-- set the script as the last step, so it doesn't fire yet
|
||||||
|
scrollbar:SetScript("OnValueChanged", ScrollBar_OnScrollValueChanged)
|
||||||
|
|
||||||
|
local scrollbg = scrollbar:CreateTexture(nil, "BACKGROUND")
|
||||||
|
scrollbg:SetAllPoints(scrollbar)
|
||||||
|
scrollbg:SetColorTexture(0, 0, 0, 0.4)
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame", nil, scrollframe)
|
||||||
|
content:SetPoint("TOPLEFT")
|
||||||
|
content:SetPoint("TOPRIGHT")
|
||||||
|
content:SetHeight(400)
|
||||||
|
scrollframe:SetScrollChild(content)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
localstatus = { scrollvalue = 0 },
|
||||||
|
scrollframe = scrollframe,
|
||||||
|
scrollbar = scrollbar,
|
||||||
|
content = content,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
scrollframe.obj, scrollbar.obj = widget, widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,69 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
SimpleGroup Container
|
||||||
|
Simple container widget that just groups widgets.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "SimpleGroup", 20
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetWidth(300)
|
||||||
|
self:SetHeight(100)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["LayoutFinished"] = function(self, width, height)
|
||||||
|
if self.noAutoHeight then return end
|
||||||
|
self:SetHeight(height or 0)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
content:SetWidth(width)
|
||||||
|
content.width = width
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
content:SetHeight(height)
|
||||||
|
content.height = height
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame", nil, frame)
|
||||||
|
content:SetPoint("TOPLEFT")
|
||||||
|
content:SetPoint("BOTTOMRIGHT")
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
frame = frame,
|
||||||
|
content = content,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,349 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
TabGroup Container
|
||||||
|
Container that uses tabs on top to switch between groups.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "TabGroup", 36
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs, ipairs, assert, type, wipe = pairs, ipairs, assert, type, wipe
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
local _G = _G
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: PanelTemplates_TabResize, PanelTemplates_SetDisabledTabState, PanelTemplates_SelectTab, PanelTemplates_DeselectTab
|
||||||
|
|
||||||
|
-- local upvalue storage used by BuildTabs
|
||||||
|
local widths = {}
|
||||||
|
local rowwidths = {}
|
||||||
|
local rowends = {}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function UpdateTabLook(frame)
|
||||||
|
if frame.disabled then
|
||||||
|
PanelTemplates_SetDisabledTabState(frame)
|
||||||
|
elseif frame.selected then
|
||||||
|
PanelTemplates_SelectTab(frame)
|
||||||
|
else
|
||||||
|
PanelTemplates_DeselectTab(frame)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tab_SetText(frame, text)
|
||||||
|
frame:_SetText(text)
|
||||||
|
local width = frame.obj.frame.width or frame.obj.frame:GetWidth() or 0
|
||||||
|
PanelTemplates_TabResize(frame, 0, nil, nil, width, frame:GetFontString():GetStringWidth())
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tab_SetSelected(frame, selected)
|
||||||
|
frame.selected = selected
|
||||||
|
UpdateTabLook(frame)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tab_SetDisabled(frame, disabled)
|
||||||
|
frame.disabled = disabled
|
||||||
|
UpdateTabLook(frame)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function BuildTabsOnUpdate(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
self:BuildTabs()
|
||||||
|
frame:SetScript("OnUpdate", nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Tab_OnClick(frame)
|
||||||
|
if not (frame.selected or frame.disabled) then
|
||||||
|
PlaySound(841) -- SOUNDKIT.IG_CHARACTER_INFO_TAB
|
||||||
|
frame.obj:SelectTab(frame.value)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tab_OnEnter(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
self:Fire("OnTabEnter", self.tabs[frame.id].value, frame)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tab_OnLeave(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
self:Fire("OnTabLeave", self.tabs[frame.id].value, frame)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tab_OnShow(frame)
|
||||||
|
_G[frame:GetName().."HighlightTexture"]:SetWidth(frame:GetTextWidth() + 30)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetTitle()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnRelease"] = function(self)
|
||||||
|
self.status = nil
|
||||||
|
for k in pairs(self.localstatus) do
|
||||||
|
self.localstatus[k] = nil
|
||||||
|
end
|
||||||
|
self.tablist = nil
|
||||||
|
for _, tab in pairs(self.tabs) do
|
||||||
|
tab:Hide()
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["CreateTab"] = function(self, id)
|
||||||
|
local tabname = ("AceGUITabGroup%dTab%d"):format(self.num, id)
|
||||||
|
local tab = CreateFrame("Button", tabname, self.border, "OptionsFrameTabButtonTemplate")
|
||||||
|
tab.obj = self
|
||||||
|
tab.id = id
|
||||||
|
|
||||||
|
tab.text = _G[tabname .. "Text"]
|
||||||
|
tab.text:ClearAllPoints()
|
||||||
|
tab.text:SetPoint("LEFT", 14, -3)
|
||||||
|
tab.text:SetPoint("RIGHT", -12, -3)
|
||||||
|
|
||||||
|
tab:SetScript("OnClick", Tab_OnClick)
|
||||||
|
tab:SetScript("OnEnter", Tab_OnEnter)
|
||||||
|
tab:SetScript("OnLeave", Tab_OnLeave)
|
||||||
|
tab:SetScript("OnShow", Tab_OnShow)
|
||||||
|
|
||||||
|
tab._SetText = tab.SetText
|
||||||
|
tab.SetText = Tab_SetText
|
||||||
|
tab.SetSelected = Tab_SetSelected
|
||||||
|
tab.SetDisabled = Tab_SetDisabled
|
||||||
|
|
||||||
|
return tab
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetTitle"] = function(self, text)
|
||||||
|
self.titletext:SetText(text or "")
|
||||||
|
if text and text ~= "" then
|
||||||
|
self.alignoffset = 25
|
||||||
|
else
|
||||||
|
self.alignoffset = 18
|
||||||
|
end
|
||||||
|
self:BuildTabs()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetStatusTable"] = function(self, status)
|
||||||
|
assert(type(status) == "table")
|
||||||
|
self.status = status
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SelectTab"] = function(self, value)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local found
|
||||||
|
for i, v in ipairs(self.tabs) do
|
||||||
|
if v.value == value then
|
||||||
|
v:SetSelected(true)
|
||||||
|
found = true
|
||||||
|
else
|
||||||
|
v:SetSelected(false)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
status.selected = value
|
||||||
|
if found then
|
||||||
|
self:Fire("OnGroupSelected",value)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetTabs"] = function(self, tabs)
|
||||||
|
self.tablist = tabs
|
||||||
|
self:BuildTabs()
|
||||||
|
end,
|
||||||
|
|
||||||
|
|
||||||
|
["BuildTabs"] = function(self)
|
||||||
|
local hastitle = (self.titletext:GetText() and self.titletext:GetText() ~= "")
|
||||||
|
local tablist = self.tablist
|
||||||
|
local tabs = self.tabs
|
||||||
|
|
||||||
|
if not tablist then return end
|
||||||
|
|
||||||
|
local width = self.frame.width or self.frame:GetWidth() or 0
|
||||||
|
|
||||||
|
wipe(widths)
|
||||||
|
wipe(rowwidths)
|
||||||
|
wipe(rowends)
|
||||||
|
|
||||||
|
--Place Text into tabs and get thier initial width
|
||||||
|
for i, v in ipairs(tablist) do
|
||||||
|
local tab = tabs[i]
|
||||||
|
if not tab then
|
||||||
|
tab = self:CreateTab(i)
|
||||||
|
tabs[i] = tab
|
||||||
|
end
|
||||||
|
|
||||||
|
tab:Show()
|
||||||
|
tab:SetText(v.text)
|
||||||
|
tab:SetDisabled(v.disabled)
|
||||||
|
tab.value = v.value
|
||||||
|
|
||||||
|
widths[i] = tab:GetWidth() - 6 --tabs are anchored 10 pixels from the right side of the previous one to reduce spacing, but add a fixed 4px padding for the text
|
||||||
|
end
|
||||||
|
|
||||||
|
for i = (#tablist)+1, #tabs, 1 do
|
||||||
|
tabs[i]:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
--First pass, find the minimum number of rows needed to hold all tabs and the initial tab layout
|
||||||
|
local numtabs = #tablist
|
||||||
|
local numrows = 1
|
||||||
|
local usedwidth = 0
|
||||||
|
|
||||||
|
for i = 1, #tablist do
|
||||||
|
--If this is not the first tab of a row and there isn't room for it
|
||||||
|
if usedwidth ~= 0 and (width - usedwidth - widths[i]) < 0 then
|
||||||
|
rowwidths[numrows] = usedwidth + 10 --first tab in each row takes up an extra 10px
|
||||||
|
rowends[numrows] = i - 1
|
||||||
|
numrows = numrows + 1
|
||||||
|
usedwidth = 0
|
||||||
|
end
|
||||||
|
usedwidth = usedwidth + widths[i]
|
||||||
|
end
|
||||||
|
rowwidths[numrows] = usedwidth + 10 --first tab in each row takes up an extra 10px
|
||||||
|
rowends[numrows] = #tablist
|
||||||
|
|
||||||
|
--Fix for single tabs being left on the last row, move a tab from the row above if applicable
|
||||||
|
if numrows > 1 then
|
||||||
|
--if the last row has only one tab
|
||||||
|
if rowends[numrows-1] == numtabs-1 then
|
||||||
|
--if there are more than 2 tabs in the 2nd last row
|
||||||
|
if (numrows == 2 and rowends[numrows-1] > 2) or (rowends[numrows] - rowends[numrows-1] > 2) then
|
||||||
|
--move 1 tab from the second last row to the last, if there is enough space
|
||||||
|
if (rowwidths[numrows] + widths[numtabs-1]) <= width then
|
||||||
|
rowends[numrows-1] = rowends[numrows-1] - 1
|
||||||
|
rowwidths[numrows] = rowwidths[numrows] + widths[numtabs-1]
|
||||||
|
rowwidths[numrows-1] = rowwidths[numrows-1] - widths[numtabs-1]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--anchor the rows as defined and resize tabs to fill thier row
|
||||||
|
local starttab = 1
|
||||||
|
for row, endtab in ipairs(rowends) do
|
||||||
|
local first = true
|
||||||
|
for tabno = starttab, endtab do
|
||||||
|
local tab = tabs[tabno]
|
||||||
|
tab:ClearAllPoints()
|
||||||
|
if first then
|
||||||
|
tab:SetPoint("TOPLEFT", self.frame, "TOPLEFT", 0, -(hastitle and 14 or 7)-(row-1)*20 )
|
||||||
|
first = false
|
||||||
|
else
|
||||||
|
tab:SetPoint("LEFT", tabs[tabno-1], "RIGHT", -10, 0)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- equal padding for each tab to fill the available width,
|
||||||
|
-- if the used space is above 75% already
|
||||||
|
-- the 18 pixel is the typical width of a scrollbar, so we can have a tab group inside a scrolling frame,
|
||||||
|
-- and not have the tabs jump around funny when switching between tabs that need scrolling and those that don't
|
||||||
|
local padding = 0
|
||||||
|
if not (numrows == 1 and rowwidths[1] < width*0.75 - 18) then
|
||||||
|
padding = (width - rowwidths[row]) / (endtab - starttab+1)
|
||||||
|
end
|
||||||
|
|
||||||
|
for i = starttab, endtab do
|
||||||
|
PanelTemplates_TabResize(tabs[i], padding + 4, nil, nil, width, tabs[i]:GetFontString():GetStringWidth())
|
||||||
|
end
|
||||||
|
starttab = endtab + 1
|
||||||
|
end
|
||||||
|
|
||||||
|
self.borderoffset = (hastitle and 17 or 10)+((numrows)*20)
|
||||||
|
self.border:SetPoint("TOPLEFT", 1, -self.borderoffset)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
local contentwidth = width - 60
|
||||||
|
if contentwidth < 0 then
|
||||||
|
contentwidth = 0
|
||||||
|
end
|
||||||
|
content:SetWidth(contentwidth)
|
||||||
|
content.width = contentwidth
|
||||||
|
self:BuildTabs(self)
|
||||||
|
self.frame:SetScript("OnUpdate", BuildTabsOnUpdate)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
local contentheight = height - (self.borderoffset + 23)
|
||||||
|
if contentheight < 0 then
|
||||||
|
contentheight = 0
|
||||||
|
end
|
||||||
|
content:SetHeight(contentheight)
|
||||||
|
content.height = contentheight
|
||||||
|
end,
|
||||||
|
|
||||||
|
["LayoutFinished"] = function(self, width, height)
|
||||||
|
if self.noAutoHeight then return end
|
||||||
|
self:SetHeight((height or 0) + (self.borderoffset + 23))
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local PaneBackdrop = {
|
||||||
|
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
edgeFile = "Interface\\Tooltips\\UI-Tooltip-Border",
|
||||||
|
tile = true, tileSize = 16, edgeSize = 16,
|
||||||
|
insets = { left = 3, right = 3, top = 5, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local num = AceGUI:GetNextWidgetNum(Type)
|
||||||
|
local frame = CreateFrame("Frame",nil,UIParent)
|
||||||
|
frame:SetHeight(100)
|
||||||
|
frame:SetWidth(100)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
|
||||||
|
local titletext = frame:CreateFontString(nil,"OVERLAY","GameFontNormal")
|
||||||
|
titletext:SetPoint("TOPLEFT", 14, 0)
|
||||||
|
titletext:SetPoint("TOPRIGHT", -14, 0)
|
||||||
|
titletext:SetJustifyH("LEFT")
|
||||||
|
titletext:SetHeight(18)
|
||||||
|
titletext:SetText("")
|
||||||
|
|
||||||
|
local border = CreateFrame("Frame", nil, frame)
|
||||||
|
border:SetPoint("TOPLEFT", 1, -27)
|
||||||
|
border:SetPoint("BOTTOMRIGHT", -1, 3)
|
||||||
|
border:SetBackdrop(PaneBackdrop)
|
||||||
|
border:SetBackdropColor(0.1, 0.1, 0.1, 0.5)
|
||||||
|
border:SetBackdropBorderColor(0.4, 0.4, 0.4)
|
||||||
|
|
||||||
|
local content = CreateFrame("Frame", nil, border)
|
||||||
|
content:SetPoint("TOPLEFT", 10, -7)
|
||||||
|
content:SetPoint("BOTTOMRIGHT", -10, 7)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
num = num,
|
||||||
|
frame = frame,
|
||||||
|
localstatus = {},
|
||||||
|
alignoffset = 18,
|
||||||
|
titletext = titletext,
|
||||||
|
border = border,
|
||||||
|
borderoffset = 27,
|
||||||
|
tabs = {},
|
||||||
|
content = content,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,718 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
TreeGroup Container
|
||||||
|
Container that uses a tree control to switch between groups.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "TreeGroup", 44
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
local WoW80 = select(4, GetBuildInfo()) >= 80000
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local next, pairs, ipairs, assert, type = next, pairs, ipairs, assert, type
|
||||||
|
local math_min, math_max, floor = math.min, math.max, floor
|
||||||
|
local select, tremove, unpack, tconcat = select, table.remove, unpack, table.concat
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: FONT_COLOR_CODE_CLOSE
|
||||||
|
|
||||||
|
-- Recycling functions
|
||||||
|
local new, del
|
||||||
|
do
|
||||||
|
local pool = setmetatable({},{__mode='k'})
|
||||||
|
function new()
|
||||||
|
local t = next(pool)
|
||||||
|
if t then
|
||||||
|
pool[t] = nil
|
||||||
|
return t
|
||||||
|
else
|
||||||
|
return {}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
function del(t)
|
||||||
|
for k in pairs(t) do
|
||||||
|
t[k] = nil
|
||||||
|
end
|
||||||
|
pool[t] = true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local DEFAULT_TREE_WIDTH = 175
|
||||||
|
local DEFAULT_TREE_SIZABLE = true
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function GetButtonUniqueValue(line)
|
||||||
|
local parent = line.parent
|
||||||
|
if parent and parent.value then
|
||||||
|
return GetButtonUniqueValue(parent).."\001"..line.value
|
||||||
|
else
|
||||||
|
return line.value
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function UpdateButton(button, treeline, selected, canExpand, isExpanded)
|
||||||
|
local self = button.obj
|
||||||
|
local toggle = button.toggle
|
||||||
|
local text = treeline.text or ""
|
||||||
|
local icon = treeline.icon
|
||||||
|
local iconCoords = treeline.iconCoords
|
||||||
|
local level = treeline.level
|
||||||
|
local value = treeline.value
|
||||||
|
local uniquevalue = treeline.uniquevalue
|
||||||
|
local disabled = treeline.disabled
|
||||||
|
|
||||||
|
button.treeline = treeline
|
||||||
|
button.value = value
|
||||||
|
button.uniquevalue = uniquevalue
|
||||||
|
if selected then
|
||||||
|
button:LockHighlight()
|
||||||
|
button.selected = true
|
||||||
|
else
|
||||||
|
button:UnlockHighlight()
|
||||||
|
button.selected = false
|
||||||
|
end
|
||||||
|
button.level = level
|
||||||
|
if ( level == 1 ) then
|
||||||
|
button:SetNormalFontObject("GameFontNormal")
|
||||||
|
button:SetHighlightFontObject("GameFontHighlight")
|
||||||
|
button.text:SetPoint("LEFT", (icon and 16 or 0) + 8, 2)
|
||||||
|
else
|
||||||
|
button:SetNormalFontObject("GameFontHighlightSmall")
|
||||||
|
button:SetHighlightFontObject("GameFontHighlightSmall")
|
||||||
|
button.text:SetPoint("LEFT", (icon and 16 or 0) + 8 * level, 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
if disabled then
|
||||||
|
button:EnableMouse(false)
|
||||||
|
button.text:SetText("|cff808080"..text..FONT_COLOR_CODE_CLOSE)
|
||||||
|
else
|
||||||
|
button.text:SetText(text)
|
||||||
|
button:EnableMouse(true)
|
||||||
|
end
|
||||||
|
|
||||||
|
if icon then
|
||||||
|
button.icon:SetTexture(icon)
|
||||||
|
button.icon:SetPoint("LEFT", 8 * level, (level == 1) and 0 or 1)
|
||||||
|
else
|
||||||
|
button.icon:SetTexture(nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
if iconCoords then
|
||||||
|
button.icon:SetTexCoord(unpack(iconCoords))
|
||||||
|
else
|
||||||
|
button.icon:SetTexCoord(0, 1, 0, 1)
|
||||||
|
end
|
||||||
|
|
||||||
|
if canExpand then
|
||||||
|
if not isExpanded then
|
||||||
|
toggle:SetNormalTexture(130838) -- Interface\\Buttons\\UI-PlusButton-UP
|
||||||
|
toggle:SetPushedTexture(130836) -- Interface\\Buttons\\UI-PlusButton-DOWN
|
||||||
|
else
|
||||||
|
toggle:SetNormalTexture(130821) -- Interface\\Buttons\\UI-MinusButton-UP
|
||||||
|
toggle:SetPushedTexture(130820) -- Interface\\Buttons\\UI-MinusButton-DOWN
|
||||||
|
end
|
||||||
|
toggle:Show()
|
||||||
|
else
|
||||||
|
toggle:Hide()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ShouldDisplayLevel(tree)
|
||||||
|
local result = false
|
||||||
|
for k, v in ipairs(tree) do
|
||||||
|
if v.children == nil and v.visible ~= false then
|
||||||
|
result = true
|
||||||
|
elseif v.children then
|
||||||
|
result = result or ShouldDisplayLevel(v.children)
|
||||||
|
end
|
||||||
|
if result then return result end
|
||||||
|
end
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
local function addLine(self, v, tree, level, parent)
|
||||||
|
local line = new()
|
||||||
|
line.value = v.value
|
||||||
|
line.text = v.text
|
||||||
|
line.icon = v.icon
|
||||||
|
line.iconCoords = v.iconCoords
|
||||||
|
line.disabled = v.disabled
|
||||||
|
line.tree = tree
|
||||||
|
line.level = level
|
||||||
|
line.parent = parent
|
||||||
|
line.visible = v.visible
|
||||||
|
line.uniquevalue = GetButtonUniqueValue(line)
|
||||||
|
if v.children then
|
||||||
|
line.hasChildren = true
|
||||||
|
else
|
||||||
|
line.hasChildren = nil
|
||||||
|
end
|
||||||
|
self.lines[#self.lines+1] = line
|
||||||
|
return line
|
||||||
|
end
|
||||||
|
|
||||||
|
--fire an update after one frame to catch the treeframes height
|
||||||
|
local function FirstFrameUpdate(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
frame:SetScript("OnUpdate", nil)
|
||||||
|
self:RefreshTree(nil, true)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function BuildUniqueValue(...)
|
||||||
|
local n = select('#', ...)
|
||||||
|
if n == 1 then
|
||||||
|
return ...
|
||||||
|
else
|
||||||
|
return (...).."\001"..BuildUniqueValue(select(2,...))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Expand_OnClick(frame)
|
||||||
|
local button = frame.button
|
||||||
|
local self = button.obj
|
||||||
|
local status = (self.status or self.localstatus).groups
|
||||||
|
status[button.uniquevalue] = not status[button.uniquevalue]
|
||||||
|
self:RefreshTree()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Button_OnClick(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
self:Fire("OnClick", frame.uniquevalue, frame.selected)
|
||||||
|
if not frame.selected then
|
||||||
|
self:SetSelected(frame.uniquevalue)
|
||||||
|
frame.selected = true
|
||||||
|
frame:LockHighlight()
|
||||||
|
self:RefreshTree()
|
||||||
|
end
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Button_OnDoubleClick(button)
|
||||||
|
local self = button.obj
|
||||||
|
local status = (self.status or self.localstatus).groups
|
||||||
|
status[button.uniquevalue] = not status[button.uniquevalue]
|
||||||
|
self:RefreshTree()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Button_OnEnter(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
self:Fire("OnButtonEnter", frame.uniquevalue, frame)
|
||||||
|
|
||||||
|
if self.enabletooltips then
|
||||||
|
local tooltip = AceGUI.tooltip
|
||||||
|
tooltip:SetOwner(frame, "ANCHOR_NONE")
|
||||||
|
tooltip:ClearAllPoints()
|
||||||
|
tooltip:SetPoint("LEFT",frame,"RIGHT")
|
||||||
|
tooltip:SetText(frame.text:GetText() or "", 1, .82, 0, true)
|
||||||
|
|
||||||
|
tooltip:Show()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Button_OnLeave(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
self:Fire("OnButtonLeave", frame.uniquevalue, frame)
|
||||||
|
|
||||||
|
if self.enabletooltips then
|
||||||
|
AceGUI.tooltip:Hide()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnScrollValueChanged(frame, value)
|
||||||
|
if frame.obj.noupdate then return end
|
||||||
|
local self = frame.obj
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
status.scrollvalue = floor(value + 0.5)
|
||||||
|
self:RefreshTree()
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tree_OnSizeChanged(frame)
|
||||||
|
frame.obj:RefreshTree()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Tree_OnMouseWheel(frame, delta)
|
||||||
|
local self = frame.obj
|
||||||
|
if self.showscroll then
|
||||||
|
local scrollbar = self.scrollbar
|
||||||
|
local min, max = scrollbar:GetMinMaxValues()
|
||||||
|
local value = scrollbar:GetValue()
|
||||||
|
local newvalue = math_min(max,math_max(min,value - delta))
|
||||||
|
if value ~= newvalue then
|
||||||
|
scrollbar:SetValue(newvalue)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Dragger_OnLeave(frame)
|
||||||
|
frame:SetBackdropColor(1, 1, 1, 0)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Dragger_OnEnter(frame)
|
||||||
|
frame:SetBackdropColor(1, 1, 1, 0.8)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Dragger_OnMouseDown(frame)
|
||||||
|
local treeframe = frame:GetParent()
|
||||||
|
treeframe:StartSizing("RIGHT")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Dragger_OnMouseUp(frame)
|
||||||
|
local treeframe = frame:GetParent()
|
||||||
|
local self = treeframe.obj
|
||||||
|
local treeframeParent = treeframe:GetParent()
|
||||||
|
treeframe:StopMovingOrSizing()
|
||||||
|
--treeframe:SetScript("OnUpdate", nil)
|
||||||
|
treeframe:SetUserPlaced(false)
|
||||||
|
--Without this :GetHeight will get stuck on the current height, causing the tree contents to not resize
|
||||||
|
treeframe:SetHeight(0)
|
||||||
|
treeframe:ClearAllPoints()
|
||||||
|
treeframe:SetPoint("TOPLEFT", treeframeParent, "TOPLEFT",0,0)
|
||||||
|
treeframe:SetPoint("BOTTOMLEFT", treeframeParent, "BOTTOMLEFT",0,0)
|
||||||
|
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
status.treewidth = treeframe:GetWidth()
|
||||||
|
|
||||||
|
treeframe.obj:Fire("OnTreeResize",treeframe:GetWidth())
|
||||||
|
-- recalculate the content width
|
||||||
|
treeframe.obj:OnWidthSet(status.fullwidth)
|
||||||
|
-- update the layout of the content
|
||||||
|
treeframe.obj:DoLayout()
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetTreeWidth(DEFAULT_TREE_WIDTH, DEFAULT_TREE_SIZABLE)
|
||||||
|
self:EnableButtonTooltips(true)
|
||||||
|
self.frame:SetScript("OnUpdate", FirstFrameUpdate)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnRelease"] = function(self)
|
||||||
|
self.status = nil
|
||||||
|
self.tree = nil
|
||||||
|
self.frame:SetScript("OnUpdate", nil)
|
||||||
|
for k, v in pairs(self.localstatus) do
|
||||||
|
if k == "groups" then
|
||||||
|
for k2 in pairs(v) do
|
||||||
|
v[k2] = nil
|
||||||
|
end
|
||||||
|
else
|
||||||
|
self.localstatus[k] = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
self.localstatus.scrollvalue = 0
|
||||||
|
self.localstatus.treewidth = DEFAULT_TREE_WIDTH
|
||||||
|
self.localstatus.treesizable = DEFAULT_TREE_SIZABLE
|
||||||
|
end,
|
||||||
|
|
||||||
|
["EnableButtonTooltips"] = function(self, enable)
|
||||||
|
self.enabletooltips = enable
|
||||||
|
end,
|
||||||
|
|
||||||
|
["CreateButton"] = function(self)
|
||||||
|
local num = AceGUI:GetNextWidgetNum("TreeGroupButton")
|
||||||
|
local button = CreateFrame("Button", ("AceGUI30TreeButton%d"):format(num), self.treeframe, "OptionsListButtonTemplate")
|
||||||
|
button.obj = self
|
||||||
|
|
||||||
|
local icon = button:CreateTexture(nil, "OVERLAY")
|
||||||
|
icon:SetWidth(14)
|
||||||
|
icon:SetHeight(14)
|
||||||
|
button.icon = icon
|
||||||
|
|
||||||
|
button:SetScript("OnClick",Button_OnClick)
|
||||||
|
button:SetScript("OnDoubleClick", Button_OnDoubleClick)
|
||||||
|
button:SetScript("OnEnter",Button_OnEnter)
|
||||||
|
button:SetScript("OnLeave",Button_OnLeave)
|
||||||
|
|
||||||
|
button.toggle.button = button
|
||||||
|
button.toggle:SetScript("OnClick",Expand_OnClick)
|
||||||
|
|
||||||
|
button.text:SetHeight(14) -- Prevents text wrapping
|
||||||
|
|
||||||
|
return button
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetStatusTable"] = function(self, status)
|
||||||
|
assert(type(status) == "table")
|
||||||
|
self.status = status
|
||||||
|
if not status.groups then
|
||||||
|
status.groups = {}
|
||||||
|
end
|
||||||
|
if not status.scrollvalue then
|
||||||
|
status.scrollvalue = 0
|
||||||
|
end
|
||||||
|
if not status.treewidth then
|
||||||
|
status.treewidth = DEFAULT_TREE_WIDTH
|
||||||
|
end
|
||||||
|
if status.treesizable == nil then
|
||||||
|
status.treesizable = DEFAULT_TREE_SIZABLE
|
||||||
|
end
|
||||||
|
self:SetTreeWidth(status.treewidth,status.treesizable)
|
||||||
|
self:RefreshTree()
|
||||||
|
end,
|
||||||
|
|
||||||
|
--sets the tree to be displayed
|
||||||
|
["SetTree"] = function(self, tree, filter)
|
||||||
|
self.filter = filter
|
||||||
|
if tree then
|
||||||
|
assert(type(tree) == "table")
|
||||||
|
end
|
||||||
|
self.tree = tree
|
||||||
|
self:RefreshTree()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["BuildLevel"] = function(self, tree, level, parent)
|
||||||
|
local groups = (self.status or self.localstatus).groups
|
||||||
|
|
||||||
|
for i, v in ipairs(tree) do
|
||||||
|
if v.children then
|
||||||
|
if not self.filter or ShouldDisplayLevel(v.children) then
|
||||||
|
local line = addLine(self, v, tree, level, parent)
|
||||||
|
if groups[line.uniquevalue] then
|
||||||
|
self:BuildLevel(v.children, level+1, line)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
elseif v.visible ~= false or not self.filter then
|
||||||
|
addLine(self, v, tree, level, parent)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["RefreshTree"] = function(self,scrollToSelection,fromOnUpdate)
|
||||||
|
local buttons = self.buttons
|
||||||
|
local lines = self.lines
|
||||||
|
|
||||||
|
for i, v in ipairs(buttons) do
|
||||||
|
v:Hide()
|
||||||
|
end
|
||||||
|
while lines[1] do
|
||||||
|
local t = tremove(lines)
|
||||||
|
for k in pairs(t) do
|
||||||
|
t[k] = nil
|
||||||
|
end
|
||||||
|
del(t)
|
||||||
|
end
|
||||||
|
|
||||||
|
if not self.tree then return end
|
||||||
|
--Build the list of visible entries from the tree and status tables
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local groupstatus = status.groups
|
||||||
|
local tree = self.tree
|
||||||
|
|
||||||
|
local treeframe = self.treeframe
|
||||||
|
|
||||||
|
status.scrollToSelection = status.scrollToSelection or scrollToSelection -- needs to be cached in case the control hasn't been drawn yet (code bails out below)
|
||||||
|
|
||||||
|
self:BuildLevel(tree, 1)
|
||||||
|
|
||||||
|
local numlines = #lines
|
||||||
|
|
||||||
|
local maxlines = (floor(((self.treeframe:GetHeight()or 0) - 20 ) / 18))
|
||||||
|
if maxlines <= 0 then return end
|
||||||
|
|
||||||
|
-- workaround for lag spikes on WoW 8.0
|
||||||
|
if WoW80 and self.frame:GetParent() == UIParent and not fromOnUpdate then
|
||||||
|
self.frame:SetScript("OnUpdate", FirstFrameUpdate)
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local first, last
|
||||||
|
|
||||||
|
scrollToSelection = status.scrollToSelection
|
||||||
|
status.scrollToSelection = nil
|
||||||
|
|
||||||
|
if numlines <= maxlines then
|
||||||
|
--the whole tree fits in the frame
|
||||||
|
status.scrollvalue = 0
|
||||||
|
self:ShowScroll(false)
|
||||||
|
first, last = 1, numlines
|
||||||
|
else
|
||||||
|
self:ShowScroll(true)
|
||||||
|
--scrolling will be needed
|
||||||
|
self.noupdate = true
|
||||||
|
self.scrollbar:SetMinMaxValues(0, numlines - maxlines)
|
||||||
|
--check if we are scrolled down too far
|
||||||
|
if numlines - status.scrollvalue < maxlines then
|
||||||
|
status.scrollvalue = numlines - maxlines
|
||||||
|
end
|
||||||
|
self.noupdate = nil
|
||||||
|
first, last = status.scrollvalue+1, status.scrollvalue + maxlines
|
||||||
|
--show selection?
|
||||||
|
if scrollToSelection and status.selected then
|
||||||
|
local show
|
||||||
|
for i,line in ipairs(lines) do -- find the line number
|
||||||
|
if line.uniquevalue==status.selected then
|
||||||
|
show=i
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if not show then
|
||||||
|
-- selection was deleted or something?
|
||||||
|
elseif show>=first and show<=last then
|
||||||
|
-- all good
|
||||||
|
else
|
||||||
|
-- scrolling needed!
|
||||||
|
if show<first then
|
||||||
|
status.scrollvalue = show-1
|
||||||
|
else
|
||||||
|
status.scrollvalue = show-maxlines
|
||||||
|
end
|
||||||
|
first, last = status.scrollvalue+1, status.scrollvalue + maxlines
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if self.scrollbar:GetValue() ~= status.scrollvalue then
|
||||||
|
self.scrollbar:SetValue(status.scrollvalue)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local buttonnum = 1
|
||||||
|
for i = first, last do
|
||||||
|
local line = lines[i]
|
||||||
|
local button = buttons[buttonnum]
|
||||||
|
if not button then
|
||||||
|
button = self:CreateButton()
|
||||||
|
|
||||||
|
buttons[buttonnum] = button
|
||||||
|
button:SetParent(treeframe)
|
||||||
|
button:SetFrameLevel(treeframe:GetFrameLevel()+1)
|
||||||
|
button:ClearAllPoints()
|
||||||
|
if buttonnum == 1 then
|
||||||
|
if self.showscroll then
|
||||||
|
button:SetPoint("TOPRIGHT", -22, -10)
|
||||||
|
button:SetPoint("TOPLEFT", 0, -10)
|
||||||
|
else
|
||||||
|
button:SetPoint("TOPRIGHT", 0, -10)
|
||||||
|
button:SetPoint("TOPLEFT", 0, -10)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
button:SetPoint("TOPRIGHT", buttons[buttonnum-1], "BOTTOMRIGHT",0,0)
|
||||||
|
button:SetPoint("TOPLEFT", buttons[buttonnum-1], "BOTTOMLEFT",0,0)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
UpdateButton(button, line, status.selected == line.uniquevalue, line.hasChildren, groupstatus[line.uniquevalue] )
|
||||||
|
button:Show()
|
||||||
|
buttonnum = buttonnum + 1
|
||||||
|
end
|
||||||
|
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetSelected"] = function(self, value)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
if status.selected ~= value then
|
||||||
|
status.selected = value
|
||||||
|
self:Fire("OnGroupSelected", value)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["Select"] = function(self, uniquevalue, ...)
|
||||||
|
self.filter = false
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local groups = status.groups
|
||||||
|
local path = {...}
|
||||||
|
for i = 1, #path do
|
||||||
|
groups[tconcat(path, "\001", 1, i)] = true
|
||||||
|
end
|
||||||
|
status.selected = uniquevalue
|
||||||
|
self:RefreshTree(true)
|
||||||
|
self:Fire("OnGroupSelected", uniquevalue)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SelectByPath"] = function(self, ...)
|
||||||
|
self:Select(BuildUniqueValue(...), ...)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SelectByValue"] = function(self, uniquevalue)
|
||||||
|
self:Select(uniquevalue, ("\001"):split(uniquevalue))
|
||||||
|
end,
|
||||||
|
|
||||||
|
["ShowScroll"] = function(self, show)
|
||||||
|
self.showscroll = show
|
||||||
|
if show then
|
||||||
|
self.scrollbar:Show()
|
||||||
|
if self.buttons[1] then
|
||||||
|
self.buttons[1]:SetPoint("TOPRIGHT", self.treeframe,"TOPRIGHT",-22,-10)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
self.scrollbar:Hide()
|
||||||
|
if self.buttons[1] then
|
||||||
|
self.buttons[1]:SetPoint("TOPRIGHT", self.treeframe,"TOPRIGHT",0,-10)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
local content = self.content
|
||||||
|
local treeframe = self.treeframe
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
status.fullwidth = width
|
||||||
|
|
||||||
|
local contentwidth = width - status.treewidth - 20
|
||||||
|
if contentwidth < 0 then
|
||||||
|
contentwidth = 0
|
||||||
|
end
|
||||||
|
content:SetWidth(contentwidth)
|
||||||
|
content.width = contentwidth
|
||||||
|
|
||||||
|
local maxtreewidth = math_min(400, width - 50)
|
||||||
|
|
||||||
|
if maxtreewidth > 100 and status.treewidth > maxtreewidth then
|
||||||
|
self:SetTreeWidth(maxtreewidth, status.treesizable)
|
||||||
|
end
|
||||||
|
treeframe:SetMaxResize(maxtreewidth, 1600)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnHeightSet"] = function(self, height)
|
||||||
|
local content = self.content
|
||||||
|
local contentheight = height - 20
|
||||||
|
if contentheight < 0 then
|
||||||
|
contentheight = 0
|
||||||
|
end
|
||||||
|
content:SetHeight(contentheight)
|
||||||
|
content.height = contentheight
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetTreeWidth"] = function(self, treewidth, resizable)
|
||||||
|
if not resizable then
|
||||||
|
if type(treewidth) == 'number' then
|
||||||
|
resizable = false
|
||||||
|
elseif type(treewidth) == 'boolean' then
|
||||||
|
resizable = treewidth
|
||||||
|
treewidth = DEFAULT_TREE_WIDTH
|
||||||
|
else
|
||||||
|
resizable = false
|
||||||
|
treewidth = DEFAULT_TREE_WIDTH
|
||||||
|
end
|
||||||
|
end
|
||||||
|
self.treeframe:SetWidth(treewidth)
|
||||||
|
self.dragger:EnableMouse(resizable)
|
||||||
|
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
status.treewidth = treewidth
|
||||||
|
status.treesizable = resizable
|
||||||
|
|
||||||
|
-- recalculate the content width
|
||||||
|
if status.fullwidth then
|
||||||
|
self:OnWidthSet(status.fullwidth)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["GetTreeWidth"] = function(self)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
return status.treewidth or DEFAULT_TREE_WIDTH
|
||||||
|
end,
|
||||||
|
|
||||||
|
["LayoutFinished"] = function(self, width, height)
|
||||||
|
if self.noAutoHeight then return end
|
||||||
|
self:SetHeight((height or 0) + 20)
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local PaneBackdrop = {
|
||||||
|
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
edgeFile = "Interface\\Tooltips\\UI-Tooltip-Border",
|
||||||
|
tile = true, tileSize = 16, edgeSize = 16,
|
||||||
|
insets = { left = 3, right = 3, top = 5, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local DraggerBackdrop = {
|
||||||
|
bgFile = "Interface\\Tooltips\\UI-Tooltip-Background",
|
||||||
|
edgeFile = nil,
|
||||||
|
tile = true, tileSize = 16, edgeSize = 0,
|
||||||
|
insets = { left = 3, right = 3, top = 7, bottom = 7 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local num = AceGUI:GetNextWidgetNum(Type)
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
|
||||||
|
local treeframe = CreateFrame("Frame", nil, frame)
|
||||||
|
treeframe:SetPoint("TOPLEFT")
|
||||||
|
treeframe:SetPoint("BOTTOMLEFT")
|
||||||
|
treeframe:SetWidth(DEFAULT_TREE_WIDTH)
|
||||||
|
treeframe:EnableMouseWheel(true)
|
||||||
|
treeframe:SetBackdrop(PaneBackdrop)
|
||||||
|
treeframe:SetBackdropColor(0.1, 0.1, 0.1, 0.5)
|
||||||
|
treeframe:SetBackdropBorderColor(0.4, 0.4, 0.4)
|
||||||
|
treeframe:SetResizable(true)
|
||||||
|
treeframe:SetMinResize(100, 1)
|
||||||
|
treeframe:SetMaxResize(400, 1600)
|
||||||
|
treeframe:SetScript("OnUpdate", FirstFrameUpdate)
|
||||||
|
treeframe:SetScript("OnSizeChanged", Tree_OnSizeChanged)
|
||||||
|
treeframe:SetScript("OnMouseWheel", Tree_OnMouseWheel)
|
||||||
|
|
||||||
|
local dragger = CreateFrame("Frame", nil, treeframe)
|
||||||
|
dragger:SetWidth(8)
|
||||||
|
dragger:SetPoint("TOP", treeframe, "TOPRIGHT")
|
||||||
|
dragger:SetPoint("BOTTOM", treeframe, "BOTTOMRIGHT")
|
||||||
|
dragger:SetBackdrop(DraggerBackdrop)
|
||||||
|
dragger:SetBackdropColor(1, 1, 1, 0)
|
||||||
|
dragger:SetScript("OnEnter", Dragger_OnEnter)
|
||||||
|
dragger:SetScript("OnLeave", Dragger_OnLeave)
|
||||||
|
dragger:SetScript("OnMouseDown", Dragger_OnMouseDown)
|
||||||
|
dragger:SetScript("OnMouseUp", Dragger_OnMouseUp)
|
||||||
|
|
||||||
|
local scrollbar = CreateFrame("Slider", ("AceConfigDialogTreeGroup%dScrollBar"):format(num), treeframe, "UIPanelScrollBarTemplate")
|
||||||
|
scrollbar:SetScript("OnValueChanged", nil)
|
||||||
|
scrollbar:SetPoint("TOPRIGHT", -10, -26)
|
||||||
|
scrollbar:SetPoint("BOTTOMRIGHT", -10, 26)
|
||||||
|
scrollbar:SetMinMaxValues(0,0)
|
||||||
|
scrollbar:SetValueStep(1)
|
||||||
|
scrollbar:SetValue(0)
|
||||||
|
scrollbar:SetWidth(16)
|
||||||
|
scrollbar:SetScript("OnValueChanged", OnScrollValueChanged)
|
||||||
|
|
||||||
|
local scrollbg = scrollbar:CreateTexture(nil, "BACKGROUND")
|
||||||
|
scrollbg:SetAllPoints(scrollbar)
|
||||||
|
scrollbg:SetColorTexture(0,0,0,0.4)
|
||||||
|
|
||||||
|
local border = CreateFrame("Frame",nil,frame)
|
||||||
|
border:SetPoint("TOPLEFT", treeframe, "TOPRIGHT")
|
||||||
|
border:SetPoint("BOTTOMRIGHT")
|
||||||
|
border:SetBackdrop(PaneBackdrop)
|
||||||
|
border:SetBackdropColor(0.1, 0.1, 0.1, 0.5)
|
||||||
|
border:SetBackdropBorderColor(0.4, 0.4, 0.4)
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame", nil, border)
|
||||||
|
content:SetPoint("TOPLEFT", 10, -10)
|
||||||
|
content:SetPoint("BOTTOMRIGHT", -10, 10)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
frame = frame,
|
||||||
|
lines = {},
|
||||||
|
levels = {},
|
||||||
|
buttons = {},
|
||||||
|
hasChildren = {},
|
||||||
|
localstatus = { groups = {}, scrollvalue = 0 },
|
||||||
|
filter = false,
|
||||||
|
treeframe = treeframe,
|
||||||
|
dragger = dragger,
|
||||||
|
scrollbar = scrollbar,
|
||||||
|
border = border,
|
||||||
|
content = content,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
treeframe.obj, dragger.obj, scrollbar.obj = widget, widget, widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsContainer(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,336 @@
|
|||||||
|
local AceGUI = LibStub("AceGUI-3.0")
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs, assert, type = pairs, assert, type
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: GameFontNormal
|
||||||
|
|
||||||
|
----------------
|
||||||
|
-- Main Frame --
|
||||||
|
----------------
|
||||||
|
--[[
|
||||||
|
Events :
|
||||||
|
OnClose
|
||||||
|
|
||||||
|
]]
|
||||||
|
do
|
||||||
|
local Type = "Window"
|
||||||
|
local Version = 6
|
||||||
|
|
||||||
|
local function frameOnShow(this)
|
||||||
|
this.obj:Fire("OnShow")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function frameOnClose(this)
|
||||||
|
this.obj:Fire("OnClose")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function closeOnClick(this)
|
||||||
|
PlaySound(799) -- SOUNDKIT.GS_TITLE_OPTION_EXIT
|
||||||
|
this.obj:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function frameOnMouseDown(this)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function titleOnMouseDown(this)
|
||||||
|
this:GetParent():StartMoving()
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function frameOnMouseUp(this)
|
||||||
|
local frame = this:GetParent()
|
||||||
|
frame:StopMovingOrSizing()
|
||||||
|
local self = frame.obj
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
status.width = frame:GetWidth()
|
||||||
|
status.height = frame:GetHeight()
|
||||||
|
status.top = frame:GetTop()
|
||||||
|
status.left = frame:GetLeft()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function sizerseOnMouseDown(this)
|
||||||
|
this:GetParent():StartSizing("BOTTOMRIGHT")
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function sizersOnMouseDown(this)
|
||||||
|
this:GetParent():StartSizing("BOTTOM")
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function sizereOnMouseDown(this)
|
||||||
|
this:GetParent():StartSizing("RIGHT")
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function sizerOnMouseUp(this)
|
||||||
|
this:GetParent():StopMovingOrSizing()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function SetTitle(self,title)
|
||||||
|
self.titletext:SetText(title)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function SetStatusText(self,text)
|
||||||
|
-- self.statustext:SetText(text)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Hide(self)
|
||||||
|
self.frame:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Show(self)
|
||||||
|
self.frame:Show()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnAcquire(self)
|
||||||
|
self.frame:SetParent(UIParent)
|
||||||
|
self.frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
self:ApplyStatus()
|
||||||
|
self:EnableResize(true)
|
||||||
|
self:Show()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnRelease(self)
|
||||||
|
self.status = nil
|
||||||
|
for k in pairs(self.localstatus) do
|
||||||
|
self.localstatus[k] = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- called to set an external table to store status in
|
||||||
|
local function SetStatusTable(self, status)
|
||||||
|
assert(type(status) == "table")
|
||||||
|
self.status = status
|
||||||
|
self:ApplyStatus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ApplyStatus(self)
|
||||||
|
local status = self.status or self.localstatus
|
||||||
|
local frame = self.frame
|
||||||
|
self:SetWidth(status.width or 700)
|
||||||
|
self:SetHeight(status.height or 500)
|
||||||
|
if status.top and status.left then
|
||||||
|
frame:SetPoint("TOP",UIParent,"BOTTOM",0,status.top)
|
||||||
|
frame:SetPoint("LEFT",UIParent,"LEFT",status.left,0)
|
||||||
|
else
|
||||||
|
frame:SetPoint("CENTER",UIParent,"CENTER")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnWidthSet(self, width)
|
||||||
|
local content = self.content
|
||||||
|
local contentwidth = width - 34
|
||||||
|
if contentwidth < 0 then
|
||||||
|
contentwidth = 0
|
||||||
|
end
|
||||||
|
content:SetWidth(contentwidth)
|
||||||
|
content.width = contentwidth
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
local function OnHeightSet(self, height)
|
||||||
|
local content = self.content
|
||||||
|
local contentheight = height - 57
|
||||||
|
if contentheight < 0 then
|
||||||
|
contentheight = 0
|
||||||
|
end
|
||||||
|
content:SetHeight(contentheight)
|
||||||
|
content.height = contentheight
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EnableResize(self, state)
|
||||||
|
local func = state and "Show" or "Hide"
|
||||||
|
self.sizer_se[func](self.sizer_se)
|
||||||
|
self.sizer_s[func](self.sizer_s)
|
||||||
|
self.sizer_e[func](self.sizer_e)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame",nil,UIParent)
|
||||||
|
local self = {}
|
||||||
|
self.type = "Window"
|
||||||
|
|
||||||
|
self.Hide = Hide
|
||||||
|
self.Show = Show
|
||||||
|
self.SetTitle = SetTitle
|
||||||
|
self.OnRelease = OnRelease
|
||||||
|
self.OnAcquire = OnAcquire
|
||||||
|
self.SetStatusText = SetStatusText
|
||||||
|
self.SetStatusTable = SetStatusTable
|
||||||
|
self.ApplyStatus = ApplyStatus
|
||||||
|
self.OnWidthSet = OnWidthSet
|
||||||
|
self.OnHeightSet = OnHeightSet
|
||||||
|
self.EnableResize = EnableResize
|
||||||
|
|
||||||
|
self.localstatus = {}
|
||||||
|
|
||||||
|
self.frame = frame
|
||||||
|
frame.obj = self
|
||||||
|
frame:SetWidth(700)
|
||||||
|
frame:SetHeight(500)
|
||||||
|
frame:SetPoint("CENTER",UIParent,"CENTER",0,0)
|
||||||
|
frame:EnableMouse()
|
||||||
|
frame:SetMovable(true)
|
||||||
|
frame:SetResizable(true)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
frame:SetScript("OnMouseDown", frameOnMouseDown)
|
||||||
|
|
||||||
|
frame:SetScript("OnShow",frameOnShow)
|
||||||
|
frame:SetScript("OnHide",frameOnClose)
|
||||||
|
frame:SetMinResize(240,240)
|
||||||
|
frame:SetToplevel(true)
|
||||||
|
|
||||||
|
local titlebg = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
titlebg:SetTexture(251966) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Title-Background
|
||||||
|
titlebg:SetPoint("TOPLEFT", 9, -6)
|
||||||
|
titlebg:SetPoint("BOTTOMRIGHT", frame, "TOPRIGHT", -28, -24)
|
||||||
|
|
||||||
|
local dialogbg = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
dialogbg:SetTexture(137056) -- Interface\\Tooltips\\UI-Tooltip-Background
|
||||||
|
dialogbg:SetPoint("TOPLEFT", 8, -24)
|
||||||
|
dialogbg:SetPoint("BOTTOMRIGHT", -6, 8)
|
||||||
|
dialogbg:SetVertexColor(0, 0, 0, .75)
|
||||||
|
|
||||||
|
local topleft = frame:CreateTexture(nil, "BORDER")
|
||||||
|
topleft:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
topleft:SetWidth(64)
|
||||||
|
topleft:SetHeight(64)
|
||||||
|
topleft:SetPoint("TOPLEFT")
|
||||||
|
topleft:SetTexCoord(0.501953125, 0.625, 0, 1)
|
||||||
|
|
||||||
|
local topright = frame:CreateTexture(nil, "BORDER")
|
||||||
|
topright:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
topright:SetWidth(64)
|
||||||
|
topright:SetHeight(64)
|
||||||
|
topright:SetPoint("TOPRIGHT")
|
||||||
|
topright:SetTexCoord(0.625, 0.75, 0, 1)
|
||||||
|
|
||||||
|
local top = frame:CreateTexture(nil, "BORDER")
|
||||||
|
top:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
top:SetHeight(64)
|
||||||
|
top:SetPoint("TOPLEFT", topleft, "TOPRIGHT")
|
||||||
|
top:SetPoint("TOPRIGHT", topright, "TOPLEFT")
|
||||||
|
top:SetTexCoord(0.25, 0.369140625, 0, 1)
|
||||||
|
|
||||||
|
local bottomleft = frame:CreateTexture(nil, "BORDER")
|
||||||
|
bottomleft:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
bottomleft:SetWidth(64)
|
||||||
|
bottomleft:SetHeight(64)
|
||||||
|
bottomleft:SetPoint("BOTTOMLEFT")
|
||||||
|
bottomleft:SetTexCoord(0.751953125, 0.875, 0, 1)
|
||||||
|
|
||||||
|
local bottomright = frame:CreateTexture(nil, "BORDER")
|
||||||
|
bottomright:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
bottomright:SetWidth(64)
|
||||||
|
bottomright:SetHeight(64)
|
||||||
|
bottomright:SetPoint("BOTTOMRIGHT")
|
||||||
|
bottomright:SetTexCoord(0.875, 1, 0, 1)
|
||||||
|
|
||||||
|
local bottom = frame:CreateTexture(nil, "BORDER")
|
||||||
|
bottom:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
bottom:SetHeight(64)
|
||||||
|
bottom:SetPoint("BOTTOMLEFT", bottomleft, "BOTTOMRIGHT")
|
||||||
|
bottom:SetPoint("BOTTOMRIGHT", bottomright, "BOTTOMLEFT")
|
||||||
|
bottom:SetTexCoord(0.376953125, 0.498046875, 0, 1)
|
||||||
|
|
||||||
|
local left = frame:CreateTexture(nil, "BORDER")
|
||||||
|
left:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
left:SetWidth(64)
|
||||||
|
left:SetPoint("TOPLEFT", topleft, "BOTTOMLEFT")
|
||||||
|
left:SetPoint("BOTTOMLEFT", bottomleft, "TOPLEFT")
|
||||||
|
left:SetTexCoord(0.001953125, 0.125, 0, 1)
|
||||||
|
|
||||||
|
local right = frame:CreateTexture(nil, "BORDER")
|
||||||
|
right:SetTexture(251963) -- Interface\\PaperDollInfoFrame\\UI-GearManager-Border
|
||||||
|
right:SetWidth(64)
|
||||||
|
right:SetPoint("TOPRIGHT", topright, "BOTTOMRIGHT")
|
||||||
|
right:SetPoint("BOTTOMRIGHT", bottomright, "TOPRIGHT")
|
||||||
|
right:SetTexCoord(0.1171875, 0.2421875, 0, 1)
|
||||||
|
|
||||||
|
local close = CreateFrame("Button", nil, frame, "UIPanelCloseButton")
|
||||||
|
close:SetPoint("TOPRIGHT", 2, 1)
|
||||||
|
close:SetScript("OnClick", closeOnClick)
|
||||||
|
self.closebutton = close
|
||||||
|
close.obj = self
|
||||||
|
|
||||||
|
local titletext = frame:CreateFontString(nil, "ARTWORK")
|
||||||
|
titletext:SetFontObject(GameFontNormal)
|
||||||
|
titletext:SetPoint("TOPLEFT", 12, -8)
|
||||||
|
titletext:SetPoint("TOPRIGHT", -32, -8)
|
||||||
|
self.titletext = titletext
|
||||||
|
|
||||||
|
local title = CreateFrame("Button", nil, frame)
|
||||||
|
title:SetPoint("TOPLEFT", titlebg)
|
||||||
|
title:SetPoint("BOTTOMRIGHT", titlebg)
|
||||||
|
title:EnableMouse()
|
||||||
|
title:SetScript("OnMouseDown",titleOnMouseDown)
|
||||||
|
title:SetScript("OnMouseUp", frameOnMouseUp)
|
||||||
|
self.title = title
|
||||||
|
|
||||||
|
local sizer_se = CreateFrame("Frame",nil,frame)
|
||||||
|
sizer_se:SetPoint("BOTTOMRIGHT",frame,"BOTTOMRIGHT",0,0)
|
||||||
|
sizer_se:SetWidth(25)
|
||||||
|
sizer_se:SetHeight(25)
|
||||||
|
sizer_se:EnableMouse()
|
||||||
|
sizer_se:SetScript("OnMouseDown",sizerseOnMouseDown)
|
||||||
|
sizer_se:SetScript("OnMouseUp", sizerOnMouseUp)
|
||||||
|
self.sizer_se = sizer_se
|
||||||
|
|
||||||
|
local line1 = sizer_se:CreateTexture(nil, "BACKGROUND")
|
||||||
|
self.line1 = line1
|
||||||
|
line1:SetWidth(14)
|
||||||
|
line1:SetHeight(14)
|
||||||
|
line1:SetPoint("BOTTOMRIGHT", -8, 8)
|
||||||
|
line1:SetTexture(137057) -- Interface\\Tooltips\\UI-Tooltip-Border
|
||||||
|
local x = 0.1 * 14/17
|
||||||
|
line1:SetTexCoord(0.05 - x, 0.5, 0.05, 0.5 + x, 0.05, 0.5 - x, 0.5 + x, 0.5)
|
||||||
|
|
||||||
|
local line2 = sizer_se:CreateTexture(nil, "BACKGROUND")
|
||||||
|
self.line2 = line2
|
||||||
|
line2:SetWidth(8)
|
||||||
|
line2:SetHeight(8)
|
||||||
|
line2:SetPoint("BOTTOMRIGHT", -8, 8)
|
||||||
|
line2:SetTexture(137057) -- Interface\\Tooltips\\UI-Tooltip-Border
|
||||||
|
local x = 0.1 * 8/17
|
||||||
|
line2:SetTexCoord(0.05 - x, 0.5, 0.05, 0.5 + x, 0.05, 0.5 - x, 0.5 + x, 0.5)
|
||||||
|
|
||||||
|
local sizer_s = CreateFrame("Frame",nil,frame)
|
||||||
|
sizer_s:SetPoint("BOTTOMRIGHT",frame,"BOTTOMRIGHT",-25,0)
|
||||||
|
sizer_s:SetPoint("BOTTOMLEFT",frame,"BOTTOMLEFT",0,0)
|
||||||
|
sizer_s:SetHeight(25)
|
||||||
|
sizer_s:EnableMouse()
|
||||||
|
sizer_s:SetScript("OnMouseDown",sizersOnMouseDown)
|
||||||
|
sizer_s:SetScript("OnMouseUp", sizerOnMouseUp)
|
||||||
|
self.sizer_s = sizer_s
|
||||||
|
|
||||||
|
local sizer_e = CreateFrame("Frame",nil,frame)
|
||||||
|
sizer_e:SetPoint("BOTTOMRIGHT",frame,"BOTTOMRIGHT",0,25)
|
||||||
|
sizer_e:SetPoint("TOPRIGHT",frame,"TOPRIGHT",0,0)
|
||||||
|
sizer_e:SetWidth(25)
|
||||||
|
sizer_e:EnableMouse()
|
||||||
|
sizer_e:SetScript("OnMouseDown",sizereOnMouseDown)
|
||||||
|
sizer_e:SetScript("OnMouseUp", sizerOnMouseUp)
|
||||||
|
self.sizer_e = sizer_e
|
||||||
|
|
||||||
|
--Container Support
|
||||||
|
local content = CreateFrame("Frame",nil,frame)
|
||||||
|
self.content = content
|
||||||
|
content.obj = self
|
||||||
|
content:SetPoint("TOPLEFT",frame,"TOPLEFT",12,-32)
|
||||||
|
content:SetPoint("BOTTOMRIGHT",frame,"BOTTOMRIGHT",-12,13)
|
||||||
|
|
||||||
|
AceGUI:RegisterAsContainer(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type,Constructor,Version)
|
||||||
|
end
|
@ -0,0 +1,103 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Button Widget
|
||||||
|
Graphical Button.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "Button", 24
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local _G = _G
|
||||||
|
local PlaySound, CreateFrame, UIParent = PlaySound, CreateFrame, UIParent
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Button_OnClick(frame, ...)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
PlaySound(852) -- SOUNDKIT.IG_MAINMENU_OPTION
|
||||||
|
frame.obj:Fire("OnClick", ...)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
-- restore default values
|
||||||
|
self:SetHeight(24)
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetDisabled(false)
|
||||||
|
self:SetAutoWidth(false)
|
||||||
|
self:SetText()
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetText"] = function(self, text)
|
||||||
|
self.text:SetText(text)
|
||||||
|
if self.autoWidth then
|
||||||
|
self:SetWidth(self.text:GetStringWidth() + 30)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetAutoWidth"] = function(self, autoWidth)
|
||||||
|
self.autoWidth = autoWidth
|
||||||
|
if self.autoWidth then
|
||||||
|
self:SetWidth(self.text:GetStringWidth() + 30)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.frame:Disable()
|
||||||
|
else
|
||||||
|
self.frame:Enable()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local name = "AceGUI30Button" .. AceGUI:GetNextWidgetNum(Type)
|
||||||
|
local frame = CreateFrame("Button", name, UIParent, "UIPanelButtonTemplate")
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
frame:EnableMouse(true)
|
||||||
|
frame:SetScript("OnClick", Button_OnClick)
|
||||||
|
frame:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
frame:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
|
||||||
|
local text = frame:GetFontString()
|
||||||
|
text:ClearAllPoints()
|
||||||
|
text:SetPoint("TOPLEFT", 15, -1)
|
||||||
|
text:SetPoint("BOTTOMRIGHT", -15, 1)
|
||||||
|
text:SetJustifyV("MIDDLE")
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
text = text,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,296 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Checkbox Widget
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "CheckBox", 26
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local select, pairs = select, pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: SetDesaturation, GameFontHighlight
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function AlignImage(self)
|
||||||
|
local img = self.image:GetTexture()
|
||||||
|
self.text:ClearAllPoints()
|
||||||
|
if not img then
|
||||||
|
self.text:SetPoint("LEFT", self.checkbg, "RIGHT")
|
||||||
|
self.text:SetPoint("RIGHT")
|
||||||
|
else
|
||||||
|
self.text:SetPoint("LEFT", self.image, "RIGHT", 1, 0)
|
||||||
|
self.text:SetPoint("RIGHT")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function CheckBox_OnMouseDown(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
if not self.disabled then
|
||||||
|
if self.image:GetTexture() then
|
||||||
|
self.text:SetPoint("LEFT", self.image,"RIGHT", 2, -1)
|
||||||
|
else
|
||||||
|
self.text:SetPoint("LEFT", self.checkbg, "RIGHT", 1, -1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function CheckBox_OnMouseUp(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
if not self.disabled then
|
||||||
|
self:ToggleChecked()
|
||||||
|
|
||||||
|
if self.checked then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
else -- for both nil and false (tristate)
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
|
||||||
|
self:Fire("OnValueChanged", self.checked)
|
||||||
|
AlignImage(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetType()
|
||||||
|
self:SetValue(false)
|
||||||
|
self:SetTriState(nil)
|
||||||
|
-- height is calculated from the width and required space for the description
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetImage()
|
||||||
|
self:SetDisabled(nil)
|
||||||
|
self:SetDescription(nil)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
if self.desc then
|
||||||
|
self.desc:SetWidth(width - 30)
|
||||||
|
if self.desc:GetText() and self.desc:GetText() ~= "" then
|
||||||
|
self:SetHeight(28 + self.desc:GetStringHeight())
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.frame:Disable()
|
||||||
|
self.text:SetTextColor(0.5, 0.5, 0.5)
|
||||||
|
SetDesaturation(self.check, true)
|
||||||
|
if self.desc then
|
||||||
|
self.desc:SetTextColor(0.5, 0.5, 0.5)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
self.frame:Enable()
|
||||||
|
self.text:SetTextColor(1, 1, 1)
|
||||||
|
if self.tristate and self.checked == nil then
|
||||||
|
SetDesaturation(self.check, true)
|
||||||
|
else
|
||||||
|
SetDesaturation(self.check, false)
|
||||||
|
end
|
||||||
|
if self.desc then
|
||||||
|
self.desc:SetTextColor(1, 1, 1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetValue"] = function(self, value)
|
||||||
|
local check = self.check
|
||||||
|
self.checked = value
|
||||||
|
if value then
|
||||||
|
SetDesaturation(check, false)
|
||||||
|
check:Show()
|
||||||
|
else
|
||||||
|
--Nil is the unknown tristate value
|
||||||
|
if self.tristate and value == nil then
|
||||||
|
SetDesaturation(check, true)
|
||||||
|
check:Show()
|
||||||
|
else
|
||||||
|
SetDesaturation(check, false)
|
||||||
|
check:Hide()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
self:SetDisabled(self.disabled)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["GetValue"] = function(self)
|
||||||
|
return self.checked
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetTriState"] = function(self, enabled)
|
||||||
|
self.tristate = enabled
|
||||||
|
self:SetValue(self:GetValue())
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetType"] = function(self, type)
|
||||||
|
local checkbg = self.checkbg
|
||||||
|
local check = self.check
|
||||||
|
local highlight = self.highlight
|
||||||
|
|
||||||
|
local size
|
||||||
|
if type == "radio" then
|
||||||
|
size = 16
|
||||||
|
checkbg:SetTexture(130843) -- Interface\\Buttons\\UI-RadioButton
|
||||||
|
checkbg:SetTexCoord(0, 0.25, 0, 1)
|
||||||
|
check:SetTexture(130843) -- Interface\\Buttons\\UI-RadioButton
|
||||||
|
check:SetTexCoord(0.25, 0.5, 0, 1)
|
||||||
|
check:SetBlendMode("ADD")
|
||||||
|
highlight:SetTexture(130843) -- Interface\\Buttons\\UI-RadioButton
|
||||||
|
highlight:SetTexCoord(0.5, 0.75, 0, 1)
|
||||||
|
else
|
||||||
|
size = 24
|
||||||
|
checkbg:SetTexture(130755) -- Interface\\Buttons\\UI-CheckBox-Up
|
||||||
|
checkbg:SetTexCoord(0, 1, 0, 1)
|
||||||
|
check:SetTexture(130751) -- Interface\\Buttons\\UI-CheckBox-Check
|
||||||
|
check:SetTexCoord(0, 1, 0, 1)
|
||||||
|
check:SetBlendMode("BLEND")
|
||||||
|
highlight:SetTexture(130753) -- Interface\\Buttons\\UI-CheckBox-Highlight
|
||||||
|
highlight:SetTexCoord(0, 1, 0, 1)
|
||||||
|
end
|
||||||
|
checkbg:SetHeight(size)
|
||||||
|
checkbg:SetWidth(size)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["ToggleChecked"] = function(self)
|
||||||
|
local value = self:GetValue()
|
||||||
|
if self.tristate then
|
||||||
|
--cycle in true, nil, false order
|
||||||
|
if value then
|
||||||
|
self:SetValue(nil)
|
||||||
|
elseif value == nil then
|
||||||
|
self:SetValue(false)
|
||||||
|
else
|
||||||
|
self:SetValue(true)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
self:SetValue(not self:GetValue())
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetLabel"] = function(self, label)
|
||||||
|
self.text:SetText(label)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDescription"] = function(self, desc)
|
||||||
|
if desc then
|
||||||
|
if not self.desc then
|
||||||
|
local desc = self.frame:CreateFontString(nil, "OVERLAY", "GameFontHighlightSmall")
|
||||||
|
desc:ClearAllPoints()
|
||||||
|
desc:SetPoint("TOPLEFT", self.checkbg, "TOPRIGHT", 5, -21)
|
||||||
|
desc:SetWidth(self.frame.width - 30)
|
||||||
|
desc:SetPoint("RIGHT", self.frame, "RIGHT", -30, 0)
|
||||||
|
desc:SetJustifyH("LEFT")
|
||||||
|
desc:SetJustifyV("TOP")
|
||||||
|
self.desc = desc
|
||||||
|
end
|
||||||
|
self.desc:Show()
|
||||||
|
--self.text:SetFontObject(GameFontNormal)
|
||||||
|
self.desc:SetText(desc)
|
||||||
|
self:SetHeight(28 + self.desc:GetStringHeight())
|
||||||
|
else
|
||||||
|
if self.desc then
|
||||||
|
self.desc:SetText("")
|
||||||
|
self.desc:Hide()
|
||||||
|
end
|
||||||
|
--self.text:SetFontObject(GameFontHighlight)
|
||||||
|
self:SetHeight(24)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetImage"] = function(self, path, ...)
|
||||||
|
local image = self.image
|
||||||
|
image:SetTexture(path)
|
||||||
|
|
||||||
|
if image:GetTexture() then
|
||||||
|
local n = select("#", ...)
|
||||||
|
if n == 4 or n == 8 then
|
||||||
|
image:SetTexCoord(...)
|
||||||
|
else
|
||||||
|
image:SetTexCoord(0, 1, 0, 1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
AlignImage(self)
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Button", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
frame:EnableMouse(true)
|
||||||
|
frame:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
frame:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
frame:SetScript("OnMouseDown", CheckBox_OnMouseDown)
|
||||||
|
frame:SetScript("OnMouseUp", CheckBox_OnMouseUp)
|
||||||
|
|
||||||
|
local checkbg = frame:CreateTexture(nil, "ARTWORK")
|
||||||
|
checkbg:SetWidth(24)
|
||||||
|
checkbg:SetHeight(24)
|
||||||
|
checkbg:SetPoint("TOPLEFT")
|
||||||
|
checkbg:SetTexture(130755) -- Interface\\Buttons\\UI-CheckBox-Up
|
||||||
|
|
||||||
|
local check = frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
check:SetAllPoints(checkbg)
|
||||||
|
check:SetTexture(130751) -- Interface\\Buttons\\UI-CheckBox-Check
|
||||||
|
|
||||||
|
local text = frame:CreateFontString(nil, "OVERLAY", "GameFontHighlight")
|
||||||
|
text:SetJustifyH("LEFT")
|
||||||
|
text:SetHeight(18)
|
||||||
|
text:SetPoint("LEFT", checkbg, "RIGHT")
|
||||||
|
text:SetPoint("RIGHT")
|
||||||
|
|
||||||
|
local highlight = frame:CreateTexture(nil, "HIGHLIGHT")
|
||||||
|
highlight:SetTexture(130753) -- Interface\\Buttons\\UI-CheckBox-Highlight
|
||||||
|
highlight:SetBlendMode("ADD")
|
||||||
|
highlight:SetAllPoints(checkbg)
|
||||||
|
|
||||||
|
local image = frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
image:SetHeight(16)
|
||||||
|
image:SetWidth(16)
|
||||||
|
image:SetPoint("LEFT", checkbg, "RIGHT", 1, 0)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
checkbg = checkbg,
|
||||||
|
check = check,
|
||||||
|
text = text,
|
||||||
|
highlight = highlight,
|
||||||
|
image = image,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,190 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
ColorPicker Widget
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "ColorPicker", 25
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: ColorPickerFrame, OpacitySliderFrame
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function ColorCallback(self, r, g, b, a, isAlpha)
|
||||||
|
if not self.HasAlpha then
|
||||||
|
a = 1
|
||||||
|
end
|
||||||
|
self:SetColor(r, g, b, a)
|
||||||
|
if ColorPickerFrame:IsVisible() then
|
||||||
|
--colorpicker is still open
|
||||||
|
self:Fire("OnValueChanged", r, g, b, a)
|
||||||
|
else
|
||||||
|
--colorpicker is closed, color callback is first, ignore it,
|
||||||
|
--alpha callback is the final call after it closes so confirm now
|
||||||
|
if isAlpha then
|
||||||
|
self:Fire("OnValueConfirmed", r, g, b, a)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ColorSwatch_OnClick(frame)
|
||||||
|
ColorPickerFrame:Hide()
|
||||||
|
local self = frame.obj
|
||||||
|
if not self.disabled then
|
||||||
|
ColorPickerFrame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
ColorPickerFrame:SetFrameLevel(frame:GetFrameLevel() + 10)
|
||||||
|
ColorPickerFrame:SetClampedToScreen(true)
|
||||||
|
|
||||||
|
ColorPickerFrame.func = function()
|
||||||
|
local r, g, b = ColorPickerFrame:GetColorRGB()
|
||||||
|
local a = 1 - OpacitySliderFrame:GetValue()
|
||||||
|
ColorCallback(self, r, g, b, a)
|
||||||
|
end
|
||||||
|
|
||||||
|
ColorPickerFrame.hasOpacity = self.HasAlpha
|
||||||
|
ColorPickerFrame.opacityFunc = function()
|
||||||
|
local r, g, b = ColorPickerFrame:GetColorRGB()
|
||||||
|
local a = 1 - OpacitySliderFrame:GetValue()
|
||||||
|
ColorCallback(self, r, g, b, a, true)
|
||||||
|
end
|
||||||
|
|
||||||
|
local r, g, b, a = self.r, self.g, self.b, self.a
|
||||||
|
if self.HasAlpha then
|
||||||
|
ColorPickerFrame.opacity = 1 - (a or 0)
|
||||||
|
end
|
||||||
|
ColorPickerFrame:SetColorRGB(r, g, b)
|
||||||
|
|
||||||
|
ColorPickerFrame.cancelFunc = function()
|
||||||
|
ColorCallback(self, r, g, b, a, true)
|
||||||
|
end
|
||||||
|
|
||||||
|
ColorPickerFrame:Show()
|
||||||
|
end
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetHeight(24)
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetHasAlpha(false)
|
||||||
|
self:SetColor(0, 0, 0, 1)
|
||||||
|
self:SetDisabled(nil)
|
||||||
|
self:SetLabel(nil)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetLabel"] = function(self, text)
|
||||||
|
self.text:SetText(text)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetColor"] = function(self, r, g, b, a)
|
||||||
|
self.r = r
|
||||||
|
self.g = g
|
||||||
|
self.b = b
|
||||||
|
self.a = a or 1
|
||||||
|
self.colorSwatch:SetVertexColor(r, g, b, a)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetHasAlpha"] = function(self, HasAlpha)
|
||||||
|
self.HasAlpha = HasAlpha
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if self.disabled then
|
||||||
|
self.frame:Disable()
|
||||||
|
self.text:SetTextColor(0.5, 0.5, 0.5)
|
||||||
|
else
|
||||||
|
self.frame:Enable()
|
||||||
|
self.text:SetTextColor(1, 1, 1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Button", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
frame:EnableMouse(true)
|
||||||
|
frame:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
frame:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
frame:SetScript("OnClick", ColorSwatch_OnClick)
|
||||||
|
|
||||||
|
local colorSwatch = frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
colorSwatch:SetWidth(19)
|
||||||
|
colorSwatch:SetHeight(19)
|
||||||
|
colorSwatch:SetTexture(130939) -- Interface\\ChatFrame\\ChatFrameColorSwatch
|
||||||
|
colorSwatch:SetPoint("LEFT")
|
||||||
|
|
||||||
|
local texture = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
colorSwatch.background = texture
|
||||||
|
texture:SetWidth(16)
|
||||||
|
texture:SetHeight(16)
|
||||||
|
texture:SetColorTexture(1, 1, 1)
|
||||||
|
texture:SetPoint("CENTER", colorSwatch)
|
||||||
|
texture:Show()
|
||||||
|
|
||||||
|
local checkers = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
colorSwatch.checkers = checkers
|
||||||
|
checkers:SetWidth(14)
|
||||||
|
checkers:SetHeight(14)
|
||||||
|
checkers:SetTexture(188523) -- Tileset\\Generic\\Checkers
|
||||||
|
checkers:SetTexCoord(.25, 0, 0.5, .25)
|
||||||
|
checkers:SetDesaturated(true)
|
||||||
|
checkers:SetVertexColor(1, 1, 1, 0.75)
|
||||||
|
checkers:SetPoint("CENTER", colorSwatch)
|
||||||
|
checkers:Show()
|
||||||
|
|
||||||
|
local text = frame:CreateFontString(nil,"OVERLAY","GameFontHighlight")
|
||||||
|
text:SetHeight(24)
|
||||||
|
text:SetJustifyH("LEFT")
|
||||||
|
text:SetTextColor(1, 1, 1)
|
||||||
|
text:SetPoint("LEFT", colorSwatch, "RIGHT", 2, 0)
|
||||||
|
text:SetPoint("RIGHT")
|
||||||
|
|
||||||
|
--local highlight = frame:CreateTexture(nil, "HIGHLIGHT")
|
||||||
|
--highlight:SetTexture(136810) -- Interface\\QuestFrame\\UI-QuestTitleHighlight
|
||||||
|
--highlight:SetBlendMode("ADD")
|
||||||
|
--highlight:SetAllPoints(frame)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
colorSwatch = colorSwatch,
|
||||||
|
text = text,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,471 @@
|
|||||||
|
--[[ $Id: AceGUIWidget-DropDown-Items.lua 1202 2019-05-15 23:11:22Z nevcairiel $ ]]--
|
||||||
|
|
||||||
|
local AceGUI = LibStub("AceGUI-3.0")
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local select, assert = select, assert
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local CreateFrame = CreateFrame
|
||||||
|
|
||||||
|
local function fixlevels(parent,...)
|
||||||
|
local i = 1
|
||||||
|
local child = select(i, ...)
|
||||||
|
while child do
|
||||||
|
child:SetFrameLevel(parent:GetFrameLevel()+1)
|
||||||
|
fixlevels(child, child:GetChildren())
|
||||||
|
i = i + 1
|
||||||
|
child = select(i, ...)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function fixstrata(strata, parent, ...)
|
||||||
|
local i = 1
|
||||||
|
local child = select(i, ...)
|
||||||
|
parent:SetFrameStrata(strata)
|
||||||
|
while child do
|
||||||
|
fixstrata(strata, child, child:GetChildren())
|
||||||
|
i = i + 1
|
||||||
|
child = select(i, ...)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- ItemBase is the base "class" for all dropdown items.
|
||||||
|
-- Each item has to use ItemBase.Create(widgetType) to
|
||||||
|
-- create an initial 'self' value.
|
||||||
|
-- ItemBase will add common functions and ui event handlers.
|
||||||
|
-- Be sure to keep basic usage when you override functions.
|
||||||
|
|
||||||
|
local ItemBase = {
|
||||||
|
-- NOTE: The ItemBase version is added to each item's version number
|
||||||
|
-- to ensure proper updates on ItemBase changes.
|
||||||
|
-- Use at least 1000er steps.
|
||||||
|
version = 1000,
|
||||||
|
counter = 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
function ItemBase.Frame_OnEnter(this)
|
||||||
|
local self = this.obj
|
||||||
|
|
||||||
|
if self.useHighlight then
|
||||||
|
self.highlight:Show()
|
||||||
|
end
|
||||||
|
self:Fire("OnEnter")
|
||||||
|
|
||||||
|
if self.specialOnEnter then
|
||||||
|
self.specialOnEnter(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function ItemBase.Frame_OnLeave(this)
|
||||||
|
local self = this.obj
|
||||||
|
|
||||||
|
self.highlight:Hide()
|
||||||
|
self:Fire("OnLeave")
|
||||||
|
|
||||||
|
if self.specialOnLeave then
|
||||||
|
self.specialOnLeave(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported, AceGUI callback
|
||||||
|
function ItemBase.OnAcquire(self)
|
||||||
|
self.frame:SetToplevel(true)
|
||||||
|
self.frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported, AceGUI callback
|
||||||
|
function ItemBase.OnRelease(self)
|
||||||
|
self:SetDisabled(false)
|
||||||
|
self.pullout = nil
|
||||||
|
self.frame:SetParent(nil)
|
||||||
|
self.frame:ClearAllPoints()
|
||||||
|
self.frame:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
-- NOTE: this is called by a Dropdown-Pullout.
|
||||||
|
-- Do not call this method directly
|
||||||
|
function ItemBase.SetPullout(self, pullout)
|
||||||
|
self.pullout = pullout
|
||||||
|
|
||||||
|
self.frame:SetParent(nil)
|
||||||
|
self.frame:SetParent(pullout.itemFrame)
|
||||||
|
self.parent = pullout.itemFrame
|
||||||
|
fixlevels(pullout.itemFrame, pullout.itemFrame:GetChildren())
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
function ItemBase.SetText(self, text)
|
||||||
|
self.text:SetText(text or "")
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
function ItemBase.GetText(self)
|
||||||
|
return self.text:GetText()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
function ItemBase.SetPoint(self, ...)
|
||||||
|
self.frame:SetPoint(...)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
function ItemBase.Show(self)
|
||||||
|
self.frame:Show()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
function ItemBase.Hide(self)
|
||||||
|
self.frame:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
function ItemBase.SetDisabled(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.useHighlight = false
|
||||||
|
self.text:SetTextColor(.5, .5, .5)
|
||||||
|
else
|
||||||
|
self.useHighlight = true
|
||||||
|
self.text:SetTextColor(1, 1, 1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
-- NOTE: this is called by a Dropdown-Pullout.
|
||||||
|
-- Do not call this method directly
|
||||||
|
function ItemBase.SetOnLeave(self, func)
|
||||||
|
self.specialOnLeave = func
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
-- NOTE: this is called by a Dropdown-Pullout.
|
||||||
|
-- Do not call this method directly
|
||||||
|
function ItemBase.SetOnEnter(self, func)
|
||||||
|
self.specialOnEnter = func
|
||||||
|
end
|
||||||
|
|
||||||
|
function ItemBase.Create(type)
|
||||||
|
-- NOTE: Most of the following code is copied from AceGUI-3.0/Dropdown widget
|
||||||
|
local count = AceGUI:GetNextWidgetNum(type)
|
||||||
|
local frame = CreateFrame("Button", "AceGUI30DropDownItem"..count)
|
||||||
|
local self = {}
|
||||||
|
self.frame = frame
|
||||||
|
frame.obj = self
|
||||||
|
self.type = type
|
||||||
|
|
||||||
|
self.useHighlight = true
|
||||||
|
|
||||||
|
frame:SetHeight(17)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
|
||||||
|
local text = frame:CreateFontString(nil,"OVERLAY","GameFontNormalSmall")
|
||||||
|
text:SetTextColor(1,1,1)
|
||||||
|
text:SetJustifyH("LEFT")
|
||||||
|
text:SetPoint("TOPLEFT",frame,"TOPLEFT",18,0)
|
||||||
|
text:SetPoint("BOTTOMRIGHT",frame,"BOTTOMRIGHT",-8,0)
|
||||||
|
self.text = text
|
||||||
|
|
||||||
|
local highlight = frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
highlight:SetTexture(136810) -- Interface\\QuestFrame\\UI-QuestTitleHighlight
|
||||||
|
highlight:SetBlendMode("ADD")
|
||||||
|
highlight:SetHeight(14)
|
||||||
|
highlight:ClearAllPoints()
|
||||||
|
highlight:SetPoint("RIGHT",frame,"RIGHT",-3,0)
|
||||||
|
highlight:SetPoint("LEFT",frame,"LEFT",5,0)
|
||||||
|
highlight:Hide()
|
||||||
|
self.highlight = highlight
|
||||||
|
|
||||||
|
local check = frame:CreateTexture("OVERLAY")
|
||||||
|
check:SetWidth(16)
|
||||||
|
check:SetHeight(16)
|
||||||
|
check:SetPoint("LEFT",frame,"LEFT",3,-1)
|
||||||
|
check:SetTexture(130751) -- Interface\\Buttons\\UI-CheckBox-Check
|
||||||
|
check:Hide()
|
||||||
|
self.check = check
|
||||||
|
|
||||||
|
local sub = frame:CreateTexture("OVERLAY")
|
||||||
|
sub:SetWidth(16)
|
||||||
|
sub:SetHeight(16)
|
||||||
|
sub:SetPoint("RIGHT",frame,"RIGHT",-3,-1)
|
||||||
|
sub:SetTexture(130940) -- Interface\\ChatFrame\\ChatFrameExpandArrow
|
||||||
|
sub:Hide()
|
||||||
|
self.sub = sub
|
||||||
|
|
||||||
|
frame:SetScript("OnEnter", ItemBase.Frame_OnEnter)
|
||||||
|
frame:SetScript("OnLeave", ItemBase.Frame_OnLeave)
|
||||||
|
|
||||||
|
self.OnAcquire = ItemBase.OnAcquire
|
||||||
|
self.OnRelease = ItemBase.OnRelease
|
||||||
|
|
||||||
|
self.SetPullout = ItemBase.SetPullout
|
||||||
|
self.GetText = ItemBase.GetText
|
||||||
|
self.SetText = ItemBase.SetText
|
||||||
|
self.SetDisabled = ItemBase.SetDisabled
|
||||||
|
|
||||||
|
self.SetPoint = ItemBase.SetPoint
|
||||||
|
self.Show = ItemBase.Show
|
||||||
|
self.Hide = ItemBase.Hide
|
||||||
|
|
||||||
|
self.SetOnLeave = ItemBase.SetOnLeave
|
||||||
|
self.SetOnEnter = ItemBase.SetOnEnter
|
||||||
|
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Register a dummy LibStub library to retrieve the ItemBase, so other addons can use it.
|
||||||
|
local IBLib = LibStub:NewLibrary("AceGUI-3.0-DropDown-ItemBase", ItemBase.version)
|
||||||
|
if IBLib then
|
||||||
|
IBLib.GetItemBase = function() return ItemBase end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[
|
||||||
|
Template for items:
|
||||||
|
|
||||||
|
-- Item:
|
||||||
|
--
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown-Item-"
|
||||||
|
local widgetVersion = 1
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local self = ItemBase.Create(widgetType)
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion + ItemBase.version)
|
||||||
|
end
|
||||||
|
--]]
|
||||||
|
|
||||||
|
-- Item: Header
|
||||||
|
-- A single text entry.
|
||||||
|
-- Special: Different text color and no highlight
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown-Item-Header"
|
||||||
|
local widgetVersion = 1
|
||||||
|
|
||||||
|
local function OnEnter(this)
|
||||||
|
local self = this.obj
|
||||||
|
self:Fire("OnEnter")
|
||||||
|
|
||||||
|
if self.specialOnEnter then
|
||||||
|
self.specialOnEnter(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnLeave(this)
|
||||||
|
local self = this.obj
|
||||||
|
self:Fire("OnLeave")
|
||||||
|
|
||||||
|
if self.specialOnLeave then
|
||||||
|
self.specialOnLeave(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported, override
|
||||||
|
local function SetDisabled(self, disabled)
|
||||||
|
ItemBase.SetDisabled(self, disabled)
|
||||||
|
if not disabled then
|
||||||
|
self.text:SetTextColor(1, 1, 0)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local self = ItemBase.Create(widgetType)
|
||||||
|
|
||||||
|
self.SetDisabled = SetDisabled
|
||||||
|
|
||||||
|
self.frame:SetScript("OnEnter", OnEnter)
|
||||||
|
self.frame:SetScript("OnLeave", OnLeave)
|
||||||
|
|
||||||
|
self.text:SetTextColor(1, 1, 0)
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion + ItemBase.version)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Item: Execute
|
||||||
|
-- A simple button
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown-Item-Execute"
|
||||||
|
local widgetVersion = 1
|
||||||
|
|
||||||
|
local function Frame_OnClick(this, button)
|
||||||
|
local self = this.obj
|
||||||
|
if self.disabled then return end
|
||||||
|
self:Fire("OnClick")
|
||||||
|
if self.pullout then
|
||||||
|
self.pullout:Close()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local self = ItemBase.Create(widgetType)
|
||||||
|
|
||||||
|
self.frame:SetScript("OnClick", Frame_OnClick)
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion + ItemBase.version)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Item: Toggle
|
||||||
|
-- Some sort of checkbox for dropdown menus.
|
||||||
|
-- Does not close the pullout on click.
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown-Item-Toggle"
|
||||||
|
local widgetVersion = 4
|
||||||
|
|
||||||
|
local function UpdateToggle(self)
|
||||||
|
if self.value then
|
||||||
|
self.check:Show()
|
||||||
|
else
|
||||||
|
self.check:Hide()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnRelease(self)
|
||||||
|
ItemBase.OnRelease(self)
|
||||||
|
self:SetValue(nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Frame_OnClick(this, button)
|
||||||
|
local self = this.obj
|
||||||
|
if self.disabled then return end
|
||||||
|
self.value = not self.value
|
||||||
|
if self.value then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
else
|
||||||
|
PlaySound(857) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_OFF
|
||||||
|
end
|
||||||
|
UpdateToggle(self)
|
||||||
|
self:Fire("OnValueChanged", self.value)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetValue(self, value)
|
||||||
|
self.value = value
|
||||||
|
UpdateToggle(self)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function GetValue(self)
|
||||||
|
return self.value
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local self = ItemBase.Create(widgetType)
|
||||||
|
|
||||||
|
self.frame:SetScript("OnClick", Frame_OnClick)
|
||||||
|
|
||||||
|
self.SetValue = SetValue
|
||||||
|
self.GetValue = GetValue
|
||||||
|
self.OnRelease = OnRelease
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion + ItemBase.version)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Item: Menu
|
||||||
|
-- Shows a submenu on mouse over
|
||||||
|
-- Does not close the pullout on click
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown-Item-Menu"
|
||||||
|
local widgetVersion = 2
|
||||||
|
|
||||||
|
local function OnEnter(this)
|
||||||
|
local self = this.obj
|
||||||
|
self:Fire("OnEnter")
|
||||||
|
|
||||||
|
if self.specialOnEnter then
|
||||||
|
self.specialOnEnter(self)
|
||||||
|
end
|
||||||
|
|
||||||
|
self.highlight:Show()
|
||||||
|
|
||||||
|
if not self.disabled and self.submenu then
|
||||||
|
self.submenu:Open("TOPLEFT", self.frame, "TOPRIGHT", self.pullout:GetRightBorderWidth(), 0, self.frame:GetFrameLevel() + 100)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnHide(this)
|
||||||
|
local self = this.obj
|
||||||
|
if self.submenu then
|
||||||
|
self.submenu:Close()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetMenu(self, menu)
|
||||||
|
assert(menu.type == "Dropdown-Pullout")
|
||||||
|
self.submenu = menu
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function CloseMenu(self)
|
||||||
|
self.submenu:Close()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local self = ItemBase.Create(widgetType)
|
||||||
|
|
||||||
|
self.sub:Show()
|
||||||
|
|
||||||
|
self.frame:SetScript("OnEnter", OnEnter)
|
||||||
|
self.frame:SetScript("OnHide", OnHide)
|
||||||
|
|
||||||
|
self.SetMenu = SetMenu
|
||||||
|
self.CloseMenu = CloseMenu
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion + ItemBase.version)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Item: Separator
|
||||||
|
-- A single line to separate items
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown-Item-Separator"
|
||||||
|
local widgetVersion = 2
|
||||||
|
|
||||||
|
-- exported, override
|
||||||
|
local function SetDisabled(self, disabled)
|
||||||
|
ItemBase.SetDisabled(self, disabled)
|
||||||
|
self.useHighlight = false
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local self = ItemBase.Create(widgetType)
|
||||||
|
|
||||||
|
self.SetDisabled = SetDisabled
|
||||||
|
|
||||||
|
local line = self.frame:CreateTexture(nil, "OVERLAY")
|
||||||
|
line:SetHeight(1)
|
||||||
|
line:SetColorTexture(.5, .5, .5)
|
||||||
|
line:SetPoint("LEFT", self.frame, "LEFT", 10, 0)
|
||||||
|
line:SetPoint("RIGHT", self.frame, "RIGHT", -10, 0)
|
||||||
|
|
||||||
|
self.text:Hide()
|
||||||
|
|
||||||
|
self.useHighlight = false
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion + ItemBase.version)
|
||||||
|
end
|
@ -0,0 +1,745 @@
|
|||||||
|
--[[ $Id: AceGUIWidget-DropDown.lua 1209 2019-06-24 21:01:01Z nevcairiel $ ]]--
|
||||||
|
local AceGUI = LibStub("AceGUI-3.0")
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local min, max, floor = math.min, math.max, math.floor
|
||||||
|
local select, pairs, ipairs, type, tostring = select, pairs, ipairs, type, tostring
|
||||||
|
local tsort = table.sort
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local UIParent, CreateFrame = UIParent, CreateFrame
|
||||||
|
local _G = _G
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: CLOSE
|
||||||
|
|
||||||
|
local function fixlevels(parent,...)
|
||||||
|
local i = 1
|
||||||
|
local child = select(i, ...)
|
||||||
|
while child do
|
||||||
|
child:SetFrameLevel(parent:GetFrameLevel()+1)
|
||||||
|
fixlevels(child, child:GetChildren())
|
||||||
|
i = i + 1
|
||||||
|
child = select(i, ...)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function fixstrata(strata, parent, ...)
|
||||||
|
local i = 1
|
||||||
|
local child = select(i, ...)
|
||||||
|
parent:SetFrameStrata(strata)
|
||||||
|
while child do
|
||||||
|
fixstrata(strata, child, child:GetChildren())
|
||||||
|
i = i + 1
|
||||||
|
child = select(i, ...)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown-Pullout"
|
||||||
|
local widgetVersion = 3
|
||||||
|
|
||||||
|
--[[ Static data ]]--
|
||||||
|
|
||||||
|
local backdrop = {
|
||||||
|
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
edgeFile = "Interface\\DialogFrame\\UI-DialogBox-Border",
|
||||||
|
edgeSize = 32,
|
||||||
|
tileSize = 32,
|
||||||
|
tile = true,
|
||||||
|
insets = { left = 11, right = 12, top = 12, bottom = 11 },
|
||||||
|
}
|
||||||
|
local sliderBackdrop = {
|
||||||
|
bgFile = "Interface\\Buttons\\UI-SliderBar-Background",
|
||||||
|
edgeFile = "Interface\\Buttons\\UI-SliderBar-Border",
|
||||||
|
tile = true, tileSize = 8, edgeSize = 8,
|
||||||
|
insets = { left = 3, right = 3, top = 3, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local defaultWidth = 200
|
||||||
|
local defaultMaxHeight = 600
|
||||||
|
|
||||||
|
--[[ UI Event Handlers ]]--
|
||||||
|
|
||||||
|
-- HACK: This should be no part of the pullout, but there
|
||||||
|
-- is no other 'clean' way to response to any item-OnEnter
|
||||||
|
-- Used to close Submenus when an other item is entered
|
||||||
|
local function OnEnter(item)
|
||||||
|
local self = item.pullout
|
||||||
|
for k, v in ipairs(self.items) do
|
||||||
|
if v.CloseMenu and v ~= item then
|
||||||
|
v:CloseMenu()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- See the note in Constructor() for each scroll related function
|
||||||
|
local function OnMouseWheel(this, value)
|
||||||
|
this.obj:MoveScroll(value)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnScrollValueChanged(this, value)
|
||||||
|
this.obj:SetScroll(value)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnSizeChanged(this)
|
||||||
|
this.obj:FixScroll()
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[ Exported methods ]]--
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetScroll(self, value)
|
||||||
|
local status = self.scrollStatus
|
||||||
|
local frame, child = self.scrollFrame, self.itemFrame
|
||||||
|
local height, viewheight = frame:GetHeight(), child:GetHeight()
|
||||||
|
|
||||||
|
local offset
|
||||||
|
if height > viewheight then
|
||||||
|
offset = 0
|
||||||
|
else
|
||||||
|
offset = floor((viewheight - height) / 1000 * value)
|
||||||
|
end
|
||||||
|
child:ClearAllPoints()
|
||||||
|
child:SetPoint("TOPLEFT", frame, "TOPLEFT", 0, offset)
|
||||||
|
child:SetPoint("TOPRIGHT", frame, "TOPRIGHT", self.slider:IsShown() and -12 or 0, offset)
|
||||||
|
status.offset = offset
|
||||||
|
status.scrollvalue = value
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function MoveScroll(self, value)
|
||||||
|
local status = self.scrollStatus
|
||||||
|
local frame, child = self.scrollFrame, self.itemFrame
|
||||||
|
local height, viewheight = frame:GetHeight(), child:GetHeight()
|
||||||
|
|
||||||
|
if height > viewheight then
|
||||||
|
self.slider:Hide()
|
||||||
|
else
|
||||||
|
self.slider:Show()
|
||||||
|
local diff = height - viewheight
|
||||||
|
local delta = 1
|
||||||
|
if value < 0 then
|
||||||
|
delta = -1
|
||||||
|
end
|
||||||
|
self.slider:SetValue(min(max(status.scrollvalue + delta*(1000/(diff/45)),0), 1000))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function FixScroll(self)
|
||||||
|
local status = self.scrollStatus
|
||||||
|
local frame, child = self.scrollFrame, self.itemFrame
|
||||||
|
local height, viewheight = frame:GetHeight(), child:GetHeight()
|
||||||
|
local offset = status.offset or 0
|
||||||
|
|
||||||
|
if viewheight < height then
|
||||||
|
self.slider:Hide()
|
||||||
|
child:SetPoint("TOPRIGHT", frame, "TOPRIGHT", 0, offset)
|
||||||
|
self.slider:SetValue(0)
|
||||||
|
else
|
||||||
|
self.slider:Show()
|
||||||
|
local value = (offset / (viewheight - height) * 1000)
|
||||||
|
if value > 1000 then value = 1000 end
|
||||||
|
self.slider:SetValue(value)
|
||||||
|
self:SetScroll(value)
|
||||||
|
if value < 1000 then
|
||||||
|
child:ClearAllPoints()
|
||||||
|
child:SetPoint("TOPLEFT", frame, "TOPLEFT", 0, offset)
|
||||||
|
child:SetPoint("TOPRIGHT", frame, "TOPRIGHT", -12, offset)
|
||||||
|
status.offset = offset
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported, AceGUI callback
|
||||||
|
local function OnAcquire(self)
|
||||||
|
self.frame:SetParent(UIParent)
|
||||||
|
--self.itemFrame:SetToplevel(true)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported, AceGUI callback
|
||||||
|
local function OnRelease(self)
|
||||||
|
self:Clear()
|
||||||
|
self.frame:ClearAllPoints()
|
||||||
|
self.frame:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function AddItem(self, item)
|
||||||
|
self.items[#self.items + 1] = item
|
||||||
|
|
||||||
|
local h = #self.items * 16
|
||||||
|
self.itemFrame:SetHeight(h)
|
||||||
|
self.frame:SetHeight(min(h + 34, self.maxHeight)) -- +34: 20 for scrollFrame placement (10 offset) and +14 for item placement
|
||||||
|
|
||||||
|
item.frame:SetPoint("LEFT", self.itemFrame, "LEFT")
|
||||||
|
item.frame:SetPoint("RIGHT", self.itemFrame, "RIGHT")
|
||||||
|
|
||||||
|
item:SetPullout(self)
|
||||||
|
item:SetOnEnter(OnEnter)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function Open(self, point, relFrame, relPoint, x, y)
|
||||||
|
local items = self.items
|
||||||
|
local frame = self.frame
|
||||||
|
local itemFrame = self.itemFrame
|
||||||
|
|
||||||
|
frame:SetPoint(point, relFrame, relPoint, x, y)
|
||||||
|
|
||||||
|
|
||||||
|
local height = 8
|
||||||
|
for i, item in pairs(items) do
|
||||||
|
if i == 1 then
|
||||||
|
item:SetPoint("TOP", itemFrame, "TOP", 0, -2)
|
||||||
|
else
|
||||||
|
item:SetPoint("TOP", items[i-1].frame, "BOTTOM", 0, 1)
|
||||||
|
end
|
||||||
|
|
||||||
|
item:Show()
|
||||||
|
|
||||||
|
height = height + 16
|
||||||
|
end
|
||||||
|
itemFrame:SetHeight(height)
|
||||||
|
fixstrata("TOOLTIP", frame, frame:GetChildren())
|
||||||
|
frame:Show()
|
||||||
|
self:Fire("OnOpen")
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function Close(self)
|
||||||
|
self.frame:Hide()
|
||||||
|
self:Fire("OnClose")
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function Clear(self)
|
||||||
|
local items = self.items
|
||||||
|
for i, item in pairs(items) do
|
||||||
|
AceGUI:Release(item)
|
||||||
|
items[i] = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function IterateItems(self)
|
||||||
|
return ipairs(self.items)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetHideOnLeave(self, val)
|
||||||
|
self.hideOnLeave = val
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetMaxHeight(self, height)
|
||||||
|
self.maxHeight = height or defaultMaxHeight
|
||||||
|
if self.frame:GetHeight() > height then
|
||||||
|
self.frame:SetHeight(height)
|
||||||
|
elseif (self.itemFrame:GetHeight() + 34) < height then
|
||||||
|
self.frame:SetHeight(self.itemFrame:GetHeight() + 34) -- see :AddItem
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function GetRightBorderWidth(self)
|
||||||
|
return 6 + (self.slider:IsShown() and 12 or 0)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function GetLeftBorderWidth(self)
|
||||||
|
return 6
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[ Constructor ]]--
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local count = AceGUI:GetNextWidgetNum(widgetType)
|
||||||
|
local frame = CreateFrame("Frame", "AceGUI30Pullout"..count, UIParent)
|
||||||
|
local self = {}
|
||||||
|
self.count = count
|
||||||
|
self.type = widgetType
|
||||||
|
self.frame = frame
|
||||||
|
frame.obj = self
|
||||||
|
|
||||||
|
self.OnAcquire = OnAcquire
|
||||||
|
self.OnRelease = OnRelease
|
||||||
|
|
||||||
|
self.AddItem = AddItem
|
||||||
|
self.Open = Open
|
||||||
|
self.Close = Close
|
||||||
|
self.Clear = Clear
|
||||||
|
self.IterateItems = IterateItems
|
||||||
|
self.SetHideOnLeave = SetHideOnLeave
|
||||||
|
|
||||||
|
self.SetScroll = SetScroll
|
||||||
|
self.MoveScroll = MoveScroll
|
||||||
|
self.FixScroll = FixScroll
|
||||||
|
|
||||||
|
self.SetMaxHeight = SetMaxHeight
|
||||||
|
self.GetRightBorderWidth = GetRightBorderWidth
|
||||||
|
self.GetLeftBorderWidth = GetLeftBorderWidth
|
||||||
|
|
||||||
|
self.items = {}
|
||||||
|
|
||||||
|
self.scrollStatus = {
|
||||||
|
scrollvalue = 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
self.maxHeight = defaultMaxHeight
|
||||||
|
|
||||||
|
frame:SetBackdrop(backdrop)
|
||||||
|
frame:SetBackdropColor(0, 0, 0)
|
||||||
|
frame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
frame:SetClampedToScreen(true)
|
||||||
|
frame:SetWidth(defaultWidth)
|
||||||
|
frame:SetHeight(self.maxHeight)
|
||||||
|
--frame:SetToplevel(true)
|
||||||
|
|
||||||
|
-- NOTE: The whole scroll frame code is copied from the AceGUI-3.0 widget ScrollFrame
|
||||||
|
local scrollFrame = CreateFrame("ScrollFrame", nil, frame)
|
||||||
|
local itemFrame = CreateFrame("Frame", nil, scrollFrame)
|
||||||
|
|
||||||
|
self.scrollFrame = scrollFrame
|
||||||
|
self.itemFrame = itemFrame
|
||||||
|
|
||||||
|
scrollFrame.obj = self
|
||||||
|
itemFrame.obj = self
|
||||||
|
|
||||||
|
local slider = CreateFrame("Slider", "AceGUI30PulloutScrollbar"..count, scrollFrame)
|
||||||
|
slider:SetOrientation("VERTICAL")
|
||||||
|
slider:SetHitRectInsets(0, 0, -10, 0)
|
||||||
|
slider:SetBackdrop(sliderBackdrop)
|
||||||
|
slider:SetWidth(8)
|
||||||
|
slider:SetThumbTexture("Interface\\Buttons\\UI-SliderBar-Button-Vertical")
|
||||||
|
slider:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
self.slider = slider
|
||||||
|
slider.obj = self
|
||||||
|
|
||||||
|
scrollFrame:SetScrollChild(itemFrame)
|
||||||
|
scrollFrame:SetPoint("TOPLEFT", frame, "TOPLEFT", 6, -12)
|
||||||
|
scrollFrame:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT", -6, 12)
|
||||||
|
scrollFrame:EnableMouseWheel(true)
|
||||||
|
scrollFrame:SetScript("OnMouseWheel", OnMouseWheel)
|
||||||
|
scrollFrame:SetScript("OnSizeChanged", OnSizeChanged)
|
||||||
|
scrollFrame:SetToplevel(true)
|
||||||
|
scrollFrame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
|
||||||
|
itemFrame:SetPoint("TOPLEFT", scrollFrame, "TOPLEFT", 0, 0)
|
||||||
|
itemFrame:SetPoint("TOPRIGHT", scrollFrame, "TOPRIGHT", -12, 0)
|
||||||
|
itemFrame:SetHeight(400)
|
||||||
|
itemFrame:SetToplevel(true)
|
||||||
|
itemFrame:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
|
||||||
|
slider:SetPoint("TOPLEFT", scrollFrame, "TOPRIGHT", -16, 0)
|
||||||
|
slider:SetPoint("BOTTOMLEFT", scrollFrame, "BOTTOMRIGHT", -16, 0)
|
||||||
|
slider:SetScript("OnValueChanged", OnScrollValueChanged)
|
||||||
|
slider:SetMinMaxValues(0, 1000)
|
||||||
|
slider:SetValueStep(1)
|
||||||
|
slider:SetValue(0)
|
||||||
|
|
||||||
|
scrollFrame:Show()
|
||||||
|
itemFrame:Show()
|
||||||
|
slider:Hide()
|
||||||
|
|
||||||
|
self:FixScroll()
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion)
|
||||||
|
end
|
||||||
|
|
||||||
|
do
|
||||||
|
local widgetType = "Dropdown"
|
||||||
|
local widgetVersion = 34
|
||||||
|
|
||||||
|
--[[ Static data ]]--
|
||||||
|
|
||||||
|
--[[ UI event handler ]]--
|
||||||
|
|
||||||
|
local function Control_OnEnter(this)
|
||||||
|
this.obj.button:LockHighlight()
|
||||||
|
this.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(this)
|
||||||
|
this.obj.button:UnlockHighlight()
|
||||||
|
this.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Dropdown_OnHide(this)
|
||||||
|
local self = this.obj
|
||||||
|
if self.open then
|
||||||
|
self.pullout:Close()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Dropdown_TogglePullout(this)
|
||||||
|
local self = this.obj
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
if self.open then
|
||||||
|
self.open = nil
|
||||||
|
self.pullout:Close()
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
else
|
||||||
|
self.open = true
|
||||||
|
self.pullout:SetWidth(self.pulloutWidth or self.frame:GetWidth())
|
||||||
|
self.pullout:Open("TOPLEFT", self.frame, "BOTTOMLEFT", 0, self.label:IsShown() and -2 or 0)
|
||||||
|
AceGUI:SetFocus(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnPulloutOpen(this)
|
||||||
|
local self = this.userdata.obj
|
||||||
|
local value = self.value
|
||||||
|
|
||||||
|
if not self.multiselect then
|
||||||
|
for i, item in this:IterateItems() do
|
||||||
|
item:SetValue(item.userdata.value == value)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
self.open = true
|
||||||
|
self:Fire("OnOpened")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnPulloutClose(this)
|
||||||
|
local self = this.userdata.obj
|
||||||
|
self.open = nil
|
||||||
|
self:Fire("OnClosed")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ShowMultiText(self)
|
||||||
|
local text
|
||||||
|
for i, widget in self.pullout:IterateItems() do
|
||||||
|
if widget.type == "Dropdown-Item-Toggle" then
|
||||||
|
if widget:GetValue() then
|
||||||
|
if text then
|
||||||
|
text = text..", "..widget:GetText()
|
||||||
|
else
|
||||||
|
text = widget:GetText()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
self:SetText(text)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnItemValueChanged(this, event, checked)
|
||||||
|
local self = this.userdata.obj
|
||||||
|
|
||||||
|
if self.multiselect then
|
||||||
|
self:Fire("OnValueChanged", this.userdata.value, checked)
|
||||||
|
ShowMultiText(self)
|
||||||
|
else
|
||||||
|
if checked then
|
||||||
|
self:SetValue(this.userdata.value)
|
||||||
|
self:Fire("OnValueChanged", this.userdata.value)
|
||||||
|
else
|
||||||
|
this:SetValue(true)
|
||||||
|
end
|
||||||
|
if self.open then
|
||||||
|
self.pullout:Close()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[ Exported methods ]]--
|
||||||
|
|
||||||
|
-- exported, AceGUI callback
|
||||||
|
local function OnAcquire(self)
|
||||||
|
local pullout = AceGUI:Create("Dropdown-Pullout")
|
||||||
|
self.pullout = pullout
|
||||||
|
pullout.userdata.obj = self
|
||||||
|
pullout:SetCallback("OnClose", OnPulloutClose)
|
||||||
|
pullout:SetCallback("OnOpen", OnPulloutOpen)
|
||||||
|
self.pullout.frame:SetFrameLevel(self.frame:GetFrameLevel() + 1)
|
||||||
|
fixlevels(self.pullout.frame, self.pullout.frame:GetChildren())
|
||||||
|
|
||||||
|
self:SetHeight(44)
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetLabel()
|
||||||
|
self:SetPulloutWidth(nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported, AceGUI callback
|
||||||
|
local function OnRelease(self)
|
||||||
|
if self.open then
|
||||||
|
self.pullout:Close()
|
||||||
|
end
|
||||||
|
AceGUI:Release(self.pullout)
|
||||||
|
self.pullout = nil
|
||||||
|
|
||||||
|
self:SetText("")
|
||||||
|
self:SetDisabled(false)
|
||||||
|
self:SetMultiselect(false)
|
||||||
|
|
||||||
|
self.value = nil
|
||||||
|
self.list = nil
|
||||||
|
self.open = nil
|
||||||
|
self.hasClose = nil
|
||||||
|
|
||||||
|
self.frame:ClearAllPoints()
|
||||||
|
self.frame:Hide()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetDisabled(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.text:SetTextColor(0.5,0.5,0.5)
|
||||||
|
self.button:Disable()
|
||||||
|
self.button_cover:Disable()
|
||||||
|
self.label:SetTextColor(0.5,0.5,0.5)
|
||||||
|
else
|
||||||
|
self.button:Enable()
|
||||||
|
self.button_cover:Enable()
|
||||||
|
self.label:SetTextColor(1,.82,0)
|
||||||
|
self.text:SetTextColor(1,1,1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function ClearFocus(self)
|
||||||
|
if self.open then
|
||||||
|
self.pullout:Close()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetText(self, text)
|
||||||
|
self.text:SetText(text or "")
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetLabel(self, text)
|
||||||
|
if text and text ~= "" then
|
||||||
|
self.label:SetText(text)
|
||||||
|
self.label:Show()
|
||||||
|
self.dropdown:SetPoint("TOPLEFT",self.frame,"TOPLEFT",-15,-14)
|
||||||
|
self:SetHeight(40)
|
||||||
|
self.alignoffset = 26
|
||||||
|
else
|
||||||
|
self.label:SetText("")
|
||||||
|
self.label:Hide()
|
||||||
|
self.dropdown:SetPoint("TOPLEFT",self.frame,"TOPLEFT",-15,0)
|
||||||
|
self:SetHeight(26)
|
||||||
|
self.alignoffset = 12
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetValue(self, value)
|
||||||
|
if self.list then
|
||||||
|
self:SetText(self.list[value] or "")
|
||||||
|
end
|
||||||
|
self.value = value
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function GetValue(self)
|
||||||
|
return self.value
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetItemValue(self, item, value)
|
||||||
|
if not self.multiselect then return end
|
||||||
|
for i, widget in self.pullout:IterateItems() do
|
||||||
|
if widget.userdata.value == item then
|
||||||
|
if widget.SetValue then
|
||||||
|
widget:SetValue(value)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
ShowMultiText(self)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetItemDisabled(self, item, disabled)
|
||||||
|
for i, widget in self.pullout:IterateItems() do
|
||||||
|
if widget.userdata.value == item then
|
||||||
|
widget:SetDisabled(disabled)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function AddListItem(self, value, text, itemType)
|
||||||
|
if not itemType then itemType = "Dropdown-Item-Toggle" end
|
||||||
|
local exists = AceGUI:GetWidgetVersion(itemType)
|
||||||
|
if not exists then error(("The given item type, %q, does not exist within AceGUI-3.0"):format(tostring(itemType)), 2) end
|
||||||
|
|
||||||
|
local item = AceGUI:Create(itemType)
|
||||||
|
item:SetText(text)
|
||||||
|
item.userdata.obj = self
|
||||||
|
item.userdata.value = value
|
||||||
|
item:SetCallback("OnValueChanged", OnItemValueChanged)
|
||||||
|
self.pullout:AddItem(item)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function AddCloseButton(self)
|
||||||
|
if not self.hasClose then
|
||||||
|
local close = AceGUI:Create("Dropdown-Item-Execute")
|
||||||
|
close:SetText(CLOSE)
|
||||||
|
self.pullout:AddItem(close)
|
||||||
|
self.hasClose = true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local sortlist = {}
|
||||||
|
local function sortTbl(x,y)
|
||||||
|
local num1, num2 = tonumber(x), tonumber(y)
|
||||||
|
if num1 and num2 then -- numeric comparison, either two numbers or numeric strings
|
||||||
|
return num1 < num2
|
||||||
|
else -- compare everything else tostring'ed
|
||||||
|
return tostring(x) < tostring(y)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local function SetList(self, list, order, itemType)
|
||||||
|
self.list = list
|
||||||
|
self.pullout:Clear()
|
||||||
|
self.hasClose = nil
|
||||||
|
if not list then return end
|
||||||
|
|
||||||
|
if type(order) ~= "table" then
|
||||||
|
for v in pairs(list) do
|
||||||
|
sortlist[#sortlist + 1] = v
|
||||||
|
end
|
||||||
|
tsort(sortlist, sortTbl)
|
||||||
|
|
||||||
|
for i, key in ipairs(sortlist) do
|
||||||
|
AddListItem(self, key, list[key], itemType)
|
||||||
|
sortlist[i] = nil
|
||||||
|
end
|
||||||
|
else
|
||||||
|
for i, key in ipairs(order) do
|
||||||
|
AddListItem(self, key, list[key], itemType)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if self.multiselect then
|
||||||
|
ShowMultiText(self)
|
||||||
|
AddCloseButton(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function AddItem(self, value, text, itemType)
|
||||||
|
if self.list then
|
||||||
|
self.list[value] = text
|
||||||
|
AddListItem(self, value, text, itemType)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function SetMultiselect(self, multi)
|
||||||
|
self.multiselect = multi
|
||||||
|
if multi then
|
||||||
|
ShowMultiText(self)
|
||||||
|
AddCloseButton(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- exported
|
||||||
|
local function GetMultiselect(self)
|
||||||
|
return self.multiselect
|
||||||
|
end
|
||||||
|
|
||||||
|
local function SetPulloutWidth(self, width)
|
||||||
|
self.pulloutWidth = width
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[ Constructor ]]--
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local count = AceGUI:GetNextWidgetNum(widgetType)
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
local dropdown = CreateFrame("Frame", "AceGUI30DropDown"..count, frame, "UIDropDownMenuTemplate")
|
||||||
|
|
||||||
|
local self = {}
|
||||||
|
self.type = widgetType
|
||||||
|
self.frame = frame
|
||||||
|
self.dropdown = dropdown
|
||||||
|
self.count = count
|
||||||
|
frame.obj = self
|
||||||
|
dropdown.obj = self
|
||||||
|
|
||||||
|
self.OnRelease = OnRelease
|
||||||
|
self.OnAcquire = OnAcquire
|
||||||
|
|
||||||
|
self.ClearFocus = ClearFocus
|
||||||
|
|
||||||
|
self.SetText = SetText
|
||||||
|
self.SetValue = SetValue
|
||||||
|
self.GetValue = GetValue
|
||||||
|
self.SetList = SetList
|
||||||
|
self.SetLabel = SetLabel
|
||||||
|
self.SetDisabled = SetDisabled
|
||||||
|
self.AddItem = AddItem
|
||||||
|
self.SetMultiselect = SetMultiselect
|
||||||
|
self.GetMultiselect = GetMultiselect
|
||||||
|
self.SetItemValue = SetItemValue
|
||||||
|
self.SetItemDisabled = SetItemDisabled
|
||||||
|
self.SetPulloutWidth = SetPulloutWidth
|
||||||
|
|
||||||
|
self.alignoffset = 26
|
||||||
|
|
||||||
|
frame:SetScript("OnHide",Dropdown_OnHide)
|
||||||
|
|
||||||
|
dropdown:ClearAllPoints()
|
||||||
|
dropdown:SetPoint("TOPLEFT",frame,"TOPLEFT",-15,0)
|
||||||
|
dropdown:SetPoint("BOTTOMRIGHT",frame,"BOTTOMRIGHT",17,0)
|
||||||
|
dropdown:SetScript("OnHide", nil)
|
||||||
|
|
||||||
|
local left = _G[dropdown:GetName() .. "Left"]
|
||||||
|
local middle = _G[dropdown:GetName() .. "Middle"]
|
||||||
|
local right = _G[dropdown:GetName() .. "Right"]
|
||||||
|
|
||||||
|
middle:ClearAllPoints()
|
||||||
|
right:ClearAllPoints()
|
||||||
|
|
||||||
|
middle:SetPoint("LEFT", left, "RIGHT", 0, 0)
|
||||||
|
middle:SetPoint("RIGHT", right, "LEFT", 0, 0)
|
||||||
|
right:SetPoint("TOPRIGHT", dropdown, "TOPRIGHT", 0, 17)
|
||||||
|
|
||||||
|
local button = _G[dropdown:GetName() .. "Button"]
|
||||||
|
self.button = button
|
||||||
|
button.obj = self
|
||||||
|
button:SetScript("OnEnter",Control_OnEnter)
|
||||||
|
button:SetScript("OnLeave",Control_OnLeave)
|
||||||
|
button:SetScript("OnClick",Dropdown_TogglePullout)
|
||||||
|
|
||||||
|
local button_cover = CreateFrame("BUTTON",nil,self.frame)
|
||||||
|
self.button_cover = button_cover
|
||||||
|
button_cover.obj = self
|
||||||
|
button_cover:SetPoint("TOPLEFT",self.frame,"BOTTOMLEFT",0,25)
|
||||||
|
button_cover:SetPoint("BOTTOMRIGHT",self.frame,"BOTTOMRIGHT")
|
||||||
|
button_cover:SetScript("OnEnter",Control_OnEnter)
|
||||||
|
button_cover:SetScript("OnLeave",Control_OnLeave)
|
||||||
|
button_cover:SetScript("OnClick",Dropdown_TogglePullout)
|
||||||
|
|
||||||
|
local text = _G[dropdown:GetName() .. "Text"]
|
||||||
|
self.text = text
|
||||||
|
text.obj = self
|
||||||
|
text:ClearAllPoints()
|
||||||
|
text:SetPoint("RIGHT", right, "RIGHT" ,-43, 2)
|
||||||
|
text:SetPoint("LEFT", left, "LEFT", 25, 2)
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil,"OVERLAY","GameFontNormalSmall")
|
||||||
|
label:SetPoint("TOPLEFT",frame,"TOPLEFT",0,0)
|
||||||
|
label:SetPoint("TOPRIGHT",frame,"TOPRIGHT",0,0)
|
||||||
|
label:SetJustifyH("LEFT")
|
||||||
|
label:SetHeight(18)
|
||||||
|
label:Hide()
|
||||||
|
self.label = label
|
||||||
|
|
||||||
|
AceGUI:RegisterAsWidget(self)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion)
|
||||||
|
end
|
@ -0,0 +1,263 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
EditBox Widget
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "EditBox", 28
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local tostring, pairs = tostring, pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local GetCursorInfo, ClearCursor, GetSpellInfo = GetCursorInfo, ClearCursor, GetSpellInfo
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
local _G = _G
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: AceGUIEditBoxInsertLink, ChatFontNormal, OKAY
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
if not AceGUIEditBoxInsertLink then
|
||||||
|
-- upgradeable hook
|
||||||
|
hooksecurefunc("ChatEdit_InsertLink", function(...) return _G.AceGUIEditBoxInsertLink(...) end)
|
||||||
|
end
|
||||||
|
|
||||||
|
function _G.AceGUIEditBoxInsertLink(text)
|
||||||
|
for i = 1, AceGUI:GetWidgetCount(Type) do
|
||||||
|
local editbox = _G["AceGUI-3.0EditBox"..i]
|
||||||
|
if editbox and editbox:IsVisible() and editbox:HasFocus() then
|
||||||
|
editbox:Insert(text)
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ShowButton(self)
|
||||||
|
if not self.disablebutton then
|
||||||
|
self.button:Show()
|
||||||
|
self.editbox:SetTextInsets(0, 20, 3, 3)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function HideButton(self)
|
||||||
|
self.button:Hide()
|
||||||
|
self.editbox:SetTextInsets(0, 0, 3, 3)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Frame_OnShowFocus(frame)
|
||||||
|
frame.obj.editbox:SetFocus()
|
||||||
|
frame:SetScript("OnShow", nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnEscapePressed(frame)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnEnterPressed(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
local value = frame:GetText()
|
||||||
|
local cancel = self:Fire("OnEnterPressed", value)
|
||||||
|
if not cancel then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
HideButton(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnReceiveDrag(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
local type, id, info = GetCursorInfo()
|
||||||
|
local name
|
||||||
|
if type == "item" then
|
||||||
|
name = info
|
||||||
|
elseif type == "spell" then
|
||||||
|
name = GetSpellInfo(id, info)
|
||||||
|
elseif type == "macro" then
|
||||||
|
name = GetMacroInfo(id)
|
||||||
|
end
|
||||||
|
if name then
|
||||||
|
self:SetText(name)
|
||||||
|
self:Fire("OnEnterPressed", name)
|
||||||
|
ClearCursor()
|
||||||
|
HideButton(self)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnTextChanged(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
local value = frame:GetText()
|
||||||
|
if tostring(value) ~= tostring(self.lasttext) then
|
||||||
|
self:Fire("OnTextChanged", value)
|
||||||
|
self.lasttext = value
|
||||||
|
ShowButton(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnFocusGained(frame)
|
||||||
|
AceGUI:SetFocus(frame.obj)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Button_OnClick(frame)
|
||||||
|
local editbox = frame.obj.editbox
|
||||||
|
editbox:ClearFocus()
|
||||||
|
EditBox_OnEnterPressed(editbox)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
-- height is controlled by SetLabel
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetDisabled(false)
|
||||||
|
self:SetLabel()
|
||||||
|
self:SetText()
|
||||||
|
self:DisableButton(false)
|
||||||
|
self:SetMaxLetters(0)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnRelease"] = function(self)
|
||||||
|
self:ClearFocus()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.editbox:EnableMouse(false)
|
||||||
|
self.editbox:ClearFocus()
|
||||||
|
self.editbox:SetTextColor(0.5,0.5,0.5)
|
||||||
|
self.label:SetTextColor(0.5,0.5,0.5)
|
||||||
|
else
|
||||||
|
self.editbox:EnableMouse(true)
|
||||||
|
self.editbox:SetTextColor(1,1,1)
|
||||||
|
self.label:SetTextColor(1,.82,0)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetText"] = function(self, text)
|
||||||
|
self.lasttext = text or ""
|
||||||
|
self.editbox:SetText(text or "")
|
||||||
|
self.editbox:SetCursorPosition(0)
|
||||||
|
HideButton(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["GetText"] = function(self, text)
|
||||||
|
return self.editbox:GetText()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetLabel"] = function(self, text)
|
||||||
|
if text and text ~= "" then
|
||||||
|
self.label:SetText(text)
|
||||||
|
self.label:Show()
|
||||||
|
self.editbox:SetPoint("TOPLEFT",self.frame,"TOPLEFT",7,-18)
|
||||||
|
self:SetHeight(44)
|
||||||
|
self.alignoffset = 30
|
||||||
|
else
|
||||||
|
self.label:SetText("")
|
||||||
|
self.label:Hide()
|
||||||
|
self.editbox:SetPoint("TOPLEFT",self.frame,"TOPLEFT",7,0)
|
||||||
|
self:SetHeight(26)
|
||||||
|
self.alignoffset = 12
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["DisableButton"] = function(self, disabled)
|
||||||
|
self.disablebutton = disabled
|
||||||
|
if disabled then
|
||||||
|
HideButton(self)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetMaxLetters"] = function (self, num)
|
||||||
|
self.editbox:SetMaxLetters(num or 0)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["ClearFocus"] = function(self)
|
||||||
|
self.editbox:ClearFocus()
|
||||||
|
self.frame:SetScript("OnShow", nil)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetFocus"] = function(self)
|
||||||
|
self.editbox:SetFocus()
|
||||||
|
if not self.frame:IsShown() then
|
||||||
|
self.frame:SetScript("OnShow", Frame_OnShowFocus)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["HighlightText"] = function(self, from, to)
|
||||||
|
self.editbox:HighlightText(from, to)
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local num = AceGUI:GetNextWidgetNum(Type)
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
local editbox = CreateFrame("EditBox", "AceGUI-3.0EditBox"..num, frame, "InputBoxTemplate")
|
||||||
|
editbox:SetAutoFocus(false)
|
||||||
|
editbox:SetFontObject(ChatFontNormal)
|
||||||
|
editbox:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
editbox:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
editbox:SetScript("OnEscapePressed", EditBox_OnEscapePressed)
|
||||||
|
editbox:SetScript("OnEnterPressed", EditBox_OnEnterPressed)
|
||||||
|
editbox:SetScript("OnTextChanged", EditBox_OnTextChanged)
|
||||||
|
editbox:SetScript("OnReceiveDrag", EditBox_OnReceiveDrag)
|
||||||
|
editbox:SetScript("OnMouseDown", EditBox_OnReceiveDrag)
|
||||||
|
editbox:SetScript("OnEditFocusGained", EditBox_OnFocusGained)
|
||||||
|
editbox:SetTextInsets(0, 0, 3, 3)
|
||||||
|
editbox:SetMaxLetters(256)
|
||||||
|
editbox:SetPoint("BOTTOMLEFT", 6, 0)
|
||||||
|
editbox:SetPoint("BOTTOMRIGHT")
|
||||||
|
editbox:SetHeight(19)
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "OVERLAY", "GameFontNormalSmall")
|
||||||
|
label:SetPoint("TOPLEFT", 0, -2)
|
||||||
|
label:SetPoint("TOPRIGHT", 0, -2)
|
||||||
|
label:SetJustifyH("LEFT")
|
||||||
|
label:SetHeight(18)
|
||||||
|
|
||||||
|
local button = CreateFrame("Button", nil, editbox, "UIPanelButtonTemplate")
|
||||||
|
button:SetWidth(40)
|
||||||
|
button:SetHeight(20)
|
||||||
|
button:SetPoint("RIGHT", -2, 0)
|
||||||
|
button:SetText(OKAY)
|
||||||
|
button:SetScript("OnClick", Button_OnClick)
|
||||||
|
button:Hide()
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
alignoffset = 30,
|
||||||
|
editbox = editbox,
|
||||||
|
label = label,
|
||||||
|
button = button,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
editbox.obj, button.obj = widget, widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,78 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Heading Widget
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "Heading", 20
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetText()
|
||||||
|
self:SetFullWidth()
|
||||||
|
self:SetHeight(18)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetText"] = function(self, text)
|
||||||
|
self.label:SetText(text or "")
|
||||||
|
if text and text ~= "" then
|
||||||
|
self.left:SetPoint("RIGHT", self.label, "LEFT", -5, 0)
|
||||||
|
self.right:Show()
|
||||||
|
else
|
||||||
|
self.left:SetPoint("RIGHT", -3, 0)
|
||||||
|
self.right:Hide()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "BACKGROUND", "GameFontNormal")
|
||||||
|
label:SetPoint("TOP")
|
||||||
|
label:SetPoint("BOTTOM")
|
||||||
|
label:SetJustifyH("CENTER")
|
||||||
|
|
||||||
|
local left = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
left:SetHeight(8)
|
||||||
|
left:SetPoint("LEFT", 3, 0)
|
||||||
|
left:SetPoint("RIGHT", label, "LEFT", -5, 0)
|
||||||
|
left:SetTexture(137057) -- Interface\\Tooltips\\UI-Tooltip-Border
|
||||||
|
left:SetTexCoord(0.81, 0.94, 0.5, 1)
|
||||||
|
|
||||||
|
local right = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
right:SetHeight(8)
|
||||||
|
right:SetPoint("RIGHT", -3, 0)
|
||||||
|
right:SetPoint("LEFT", label, "RIGHT", 5, 0)
|
||||||
|
right:SetTexture(137057) -- Interface\\Tooltips\\UI-Tooltip-Border
|
||||||
|
right:SetTexCoord(0.81, 0.94, 0.5, 1)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
label = label,
|
||||||
|
left = left,
|
||||||
|
right = right,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,140 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Icon Widget
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "Icon", 21
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local select, pairs, print = select, pairs, print
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Button_OnClick(frame, button)
|
||||||
|
frame.obj:Fire("OnClick", button)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetHeight(110)
|
||||||
|
self:SetWidth(110)
|
||||||
|
self:SetLabel()
|
||||||
|
self:SetImage(nil)
|
||||||
|
self:SetImageSize(64, 64)
|
||||||
|
self:SetDisabled(false)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetLabel"] = function(self, text)
|
||||||
|
if text and text ~= "" then
|
||||||
|
self.label:Show()
|
||||||
|
self.label:SetText(text)
|
||||||
|
self:SetHeight(self.image:GetHeight() + 25)
|
||||||
|
else
|
||||||
|
self.label:Hide()
|
||||||
|
self:SetHeight(self.image:GetHeight() + 10)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetImage"] = function(self, path, ...)
|
||||||
|
local image = self.image
|
||||||
|
image:SetTexture(path)
|
||||||
|
|
||||||
|
if image:GetTexture() then
|
||||||
|
local n = select("#", ...)
|
||||||
|
if n == 4 or n == 8 then
|
||||||
|
image:SetTexCoord(...)
|
||||||
|
else
|
||||||
|
image:SetTexCoord(0, 1, 0, 1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetImageSize"] = function(self, width, height)
|
||||||
|
self.image:SetWidth(width)
|
||||||
|
self.image:SetHeight(height)
|
||||||
|
--self.frame:SetWidth(width + 30)
|
||||||
|
if self.label:IsShown() then
|
||||||
|
self:SetHeight(height + 25)
|
||||||
|
else
|
||||||
|
self:SetHeight(height + 10)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.frame:Disable()
|
||||||
|
self.label:SetTextColor(0.5, 0.5, 0.5)
|
||||||
|
self.image:SetVertexColor(0.5, 0.5, 0.5, 0.5)
|
||||||
|
else
|
||||||
|
self.frame:Enable()
|
||||||
|
self.label:SetTextColor(1, 1, 1)
|
||||||
|
self.image:SetVertexColor(1, 1, 1, 1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Button", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
frame:EnableMouse(true)
|
||||||
|
frame:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
frame:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
frame:SetScript("OnClick", Button_OnClick)
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "BACKGROUND", "GameFontHighlight")
|
||||||
|
label:SetPoint("BOTTOMLEFT")
|
||||||
|
label:SetPoint("BOTTOMRIGHT")
|
||||||
|
label:SetJustifyH("CENTER")
|
||||||
|
label:SetJustifyV("TOP")
|
||||||
|
label:SetHeight(18)
|
||||||
|
|
||||||
|
local image = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
image:SetWidth(64)
|
||||||
|
image:SetHeight(64)
|
||||||
|
image:SetPoint("TOP", 0, -5)
|
||||||
|
|
||||||
|
local highlight = frame:CreateTexture(nil, "HIGHLIGHT")
|
||||||
|
highlight:SetAllPoints(image)
|
||||||
|
highlight:SetTexture(136580) -- Interface\\PaperDollInfoFrame\\UI-Character-Tab-Highlight
|
||||||
|
highlight:SetTexCoord(0, 1, 0.23, 0.77)
|
||||||
|
highlight:SetBlendMode("ADD")
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
label = label,
|
||||||
|
image = image,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
widget.SetText = function(self, ...) print("AceGUI-3.0-Icon: SetText is deprecated! Use SetLabel instead!"); self:SetLabel(...) end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,94 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
InteractiveLabel Widget
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "InteractiveLabel", 21
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local select, pairs = select, pairs
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Label_OnClick(frame, button)
|
||||||
|
frame.obj:Fire("OnClick", button)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:LabelOnAcquire()
|
||||||
|
self:SetHighlight()
|
||||||
|
self:SetHighlightTexCoord()
|
||||||
|
self:SetDisabled(false)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetHighlight"] = function(self, ...)
|
||||||
|
self.highlight:SetTexture(...)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetHighlightTexCoord"] = function(self, ...)
|
||||||
|
local c = select("#", ...)
|
||||||
|
if c == 4 or c == 8 then
|
||||||
|
self.highlight:SetTexCoord(...)
|
||||||
|
else
|
||||||
|
self.highlight:SetTexCoord(0, 1, 0, 1)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self,disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.frame:EnableMouse(false)
|
||||||
|
self.label:SetTextColor(0.5, 0.5, 0.5)
|
||||||
|
else
|
||||||
|
self.frame:EnableMouse(true)
|
||||||
|
self.label:SetTextColor(1, 1, 1)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
-- create a Label type that we will hijack
|
||||||
|
local label = AceGUI:Create("Label")
|
||||||
|
|
||||||
|
local frame = label.frame
|
||||||
|
frame:EnableMouse(true)
|
||||||
|
frame:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
frame:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
frame:SetScript("OnMouseDown", Label_OnClick)
|
||||||
|
|
||||||
|
local highlight = frame:CreateTexture(nil, "HIGHLIGHT")
|
||||||
|
highlight:SetTexture(nil)
|
||||||
|
highlight:SetAllPoints()
|
||||||
|
highlight:SetBlendMode("ADD")
|
||||||
|
|
||||||
|
label.highlight = highlight
|
||||||
|
label.type = Type
|
||||||
|
label.LabelOnAcquire = label.OnAcquire
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
label[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return label
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
||||||
|
|
@ -0,0 +1,249 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Keybinding Widget
|
||||||
|
Set Keybindings in the Config UI.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "Keybinding", 25
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local IsShiftKeyDown, IsControlKeyDown, IsAltKeyDown = IsShiftKeyDown, IsControlKeyDown, IsAltKeyDown
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: NOT_BOUND
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Keybinding_OnClick(frame, button)
|
||||||
|
if button == "LeftButton" or button == "RightButton" then
|
||||||
|
local self = frame.obj
|
||||||
|
if self.waitingForKey then
|
||||||
|
frame:EnableKeyboard(false)
|
||||||
|
frame:EnableMouseWheel(false)
|
||||||
|
self.msgframe:Hide()
|
||||||
|
frame:UnlockHighlight()
|
||||||
|
self.waitingForKey = nil
|
||||||
|
else
|
||||||
|
frame:EnableKeyboard(true)
|
||||||
|
frame:EnableMouseWheel(true)
|
||||||
|
self.msgframe:Show()
|
||||||
|
frame:LockHighlight()
|
||||||
|
self.waitingForKey = true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local ignoreKeys = {
|
||||||
|
["BUTTON1"] = true, ["BUTTON2"] = true,
|
||||||
|
["UNKNOWN"] = true,
|
||||||
|
["LSHIFT"] = true, ["LCTRL"] = true, ["LALT"] = true,
|
||||||
|
["RSHIFT"] = true, ["RCTRL"] = true, ["RALT"] = true,
|
||||||
|
}
|
||||||
|
local function Keybinding_OnKeyDown(frame, key)
|
||||||
|
local self = frame.obj
|
||||||
|
if self.waitingForKey then
|
||||||
|
local keyPressed = key
|
||||||
|
if keyPressed == "ESCAPE" then
|
||||||
|
keyPressed = ""
|
||||||
|
else
|
||||||
|
if ignoreKeys[keyPressed] then return end
|
||||||
|
if IsShiftKeyDown() then
|
||||||
|
keyPressed = "SHIFT-"..keyPressed
|
||||||
|
end
|
||||||
|
if IsControlKeyDown() then
|
||||||
|
keyPressed = "CTRL-"..keyPressed
|
||||||
|
end
|
||||||
|
if IsAltKeyDown() then
|
||||||
|
keyPressed = "ALT-"..keyPressed
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
frame:EnableKeyboard(false)
|
||||||
|
frame:EnableMouseWheel(false)
|
||||||
|
self.msgframe:Hide()
|
||||||
|
frame:UnlockHighlight()
|
||||||
|
self.waitingForKey = nil
|
||||||
|
|
||||||
|
if not self.disabled then
|
||||||
|
self:SetKey(keyPressed)
|
||||||
|
self:Fire("OnKeyChanged", keyPressed)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Keybinding_OnMouseDown(frame, button)
|
||||||
|
if button == "LeftButton" or button == "RightButton" then
|
||||||
|
return
|
||||||
|
elseif button == "MiddleButton" then
|
||||||
|
button = "BUTTON3"
|
||||||
|
elseif button == "Button4" then
|
||||||
|
button = "BUTTON4"
|
||||||
|
elseif button == "Button5" then
|
||||||
|
button = "BUTTON5"
|
||||||
|
end
|
||||||
|
Keybinding_OnKeyDown(frame, button)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Keybinding_OnMouseWheel(frame, direction)
|
||||||
|
local button
|
||||||
|
if direction >= 0 then
|
||||||
|
button = "MOUSEWHEELUP"
|
||||||
|
else
|
||||||
|
button = "MOUSEWHEELDOWN"
|
||||||
|
end
|
||||||
|
Keybinding_OnKeyDown(frame, button)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetLabel("")
|
||||||
|
self:SetKey("")
|
||||||
|
self.waitingForKey = nil
|
||||||
|
self.msgframe:Hide()
|
||||||
|
self:SetDisabled(false)
|
||||||
|
self.button:EnableKeyboard(false)
|
||||||
|
self.button:EnableMouseWheel(false)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.button:Disable()
|
||||||
|
self.label:SetTextColor(0.5,0.5,0.5)
|
||||||
|
else
|
||||||
|
self.button:Enable()
|
||||||
|
self.label:SetTextColor(1,1,1)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetKey"] = function(self, key)
|
||||||
|
if (key or "") == "" then
|
||||||
|
self.button:SetText(NOT_BOUND)
|
||||||
|
self.button:SetNormalFontObject("GameFontNormal")
|
||||||
|
else
|
||||||
|
self.button:SetText(key)
|
||||||
|
self.button:SetNormalFontObject("GameFontHighlight")
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["GetKey"] = function(self)
|
||||||
|
local key = self.button:GetText()
|
||||||
|
if key == NOT_BOUND then
|
||||||
|
key = nil
|
||||||
|
end
|
||||||
|
return key
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetLabel"] = function(self, label)
|
||||||
|
self.label:SetText(label or "")
|
||||||
|
if (label or "") == "" then
|
||||||
|
self.alignoffset = nil
|
||||||
|
self:SetHeight(24)
|
||||||
|
else
|
||||||
|
self.alignoffset = 30
|
||||||
|
self:SetHeight(44)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
|
||||||
|
local ControlBackdrop = {
|
||||||
|
bgFile = "Interface\\Tooltips\\UI-Tooltip-Background",
|
||||||
|
edgeFile = "Interface\\Tooltips\\UI-Tooltip-Border",
|
||||||
|
tile = true, tileSize = 16, edgeSize = 16,
|
||||||
|
insets = { left = 3, right = 3, top = 3, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local function keybindingMsgFixWidth(frame)
|
||||||
|
frame:SetWidth(frame.msg:GetWidth() + 10)
|
||||||
|
frame:SetScript("OnUpdate", nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local name = "AceGUI30KeybindingButton" .. AceGUI:GetNextWidgetNum(Type)
|
||||||
|
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
local button = CreateFrame("Button", name, frame, "UIPanelButtonTemplate")
|
||||||
|
|
||||||
|
button:EnableMouse(true)
|
||||||
|
button:EnableMouseWheel(false)
|
||||||
|
button:RegisterForClicks("AnyDown")
|
||||||
|
button:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
button:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
button:SetScript("OnClick", Keybinding_OnClick)
|
||||||
|
button:SetScript("OnKeyDown", Keybinding_OnKeyDown)
|
||||||
|
button:SetScript("OnMouseDown", Keybinding_OnMouseDown)
|
||||||
|
button:SetScript("OnMouseWheel", Keybinding_OnMouseWheel)
|
||||||
|
button:SetPoint("BOTTOMLEFT")
|
||||||
|
button:SetPoint("BOTTOMRIGHT")
|
||||||
|
button:SetHeight(24)
|
||||||
|
button:EnableKeyboard(false)
|
||||||
|
|
||||||
|
local text = button:GetFontString()
|
||||||
|
text:SetPoint("LEFT", 7, 0)
|
||||||
|
text:SetPoint("RIGHT", -7, 0)
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "OVERLAY", "GameFontHighlight")
|
||||||
|
label:SetPoint("TOPLEFT")
|
||||||
|
label:SetPoint("TOPRIGHT")
|
||||||
|
label:SetJustifyH("CENTER")
|
||||||
|
label:SetHeight(18)
|
||||||
|
|
||||||
|
local msgframe = CreateFrame("Frame", nil, UIParent)
|
||||||
|
msgframe:SetHeight(30)
|
||||||
|
msgframe:SetBackdrop(ControlBackdrop)
|
||||||
|
msgframe:SetBackdropColor(0,0,0)
|
||||||
|
msgframe:SetFrameStrata("FULLSCREEN_DIALOG")
|
||||||
|
msgframe:SetFrameLevel(1000)
|
||||||
|
msgframe:SetToplevel(true)
|
||||||
|
|
||||||
|
local msg = msgframe:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||||
|
msg:SetText("Press a key to bind, ESC to clear the binding or click the button again to cancel.")
|
||||||
|
msgframe.msg = msg
|
||||||
|
msg:SetPoint("TOPLEFT", 5, -5)
|
||||||
|
msgframe:SetScript("OnUpdate", keybindingMsgFixWidth)
|
||||||
|
msgframe:SetPoint("BOTTOM", button, "TOP")
|
||||||
|
msgframe:Hide()
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
button = button,
|
||||||
|
label = label,
|
||||||
|
msgframe = msgframe,
|
||||||
|
frame = frame,
|
||||||
|
alignoffset = 30,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
button.obj = widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,178 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Label Widget
|
||||||
|
Displays text and optionally an icon.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "Label", 26
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local max, select, pairs = math.max, select, pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: GameFontHighlightSmall
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
|
||||||
|
local function UpdateImageAnchor(self)
|
||||||
|
if self.resizing then return end
|
||||||
|
local frame = self.frame
|
||||||
|
local width = frame.width or frame:GetWidth() or 0
|
||||||
|
local image = self.image
|
||||||
|
local label = self.label
|
||||||
|
local height
|
||||||
|
|
||||||
|
label:ClearAllPoints()
|
||||||
|
image:ClearAllPoints()
|
||||||
|
|
||||||
|
if self.imageshown then
|
||||||
|
local imagewidth = image:GetWidth()
|
||||||
|
if (width - imagewidth) < 200 or (label:GetText() or "") == "" then
|
||||||
|
-- image goes on top centered when less than 200 width for the text, or if there is no text
|
||||||
|
image:SetPoint("TOP")
|
||||||
|
label:SetPoint("TOP", image, "BOTTOM")
|
||||||
|
label:SetPoint("LEFT")
|
||||||
|
label:SetWidth(width)
|
||||||
|
height = image:GetHeight() + label:GetStringHeight()
|
||||||
|
else
|
||||||
|
-- image on the left
|
||||||
|
image:SetPoint("TOPLEFT")
|
||||||
|
if image:GetHeight() > label:GetStringHeight() then
|
||||||
|
label:SetPoint("LEFT", image, "RIGHT", 4, 0)
|
||||||
|
else
|
||||||
|
label:SetPoint("TOPLEFT", image, "TOPRIGHT", 4, 0)
|
||||||
|
end
|
||||||
|
label:SetWidth(width - imagewidth - 4)
|
||||||
|
height = max(image:GetHeight(), label:GetStringHeight())
|
||||||
|
end
|
||||||
|
else
|
||||||
|
-- no image shown
|
||||||
|
label:SetPoint("TOPLEFT")
|
||||||
|
label:SetWidth(width)
|
||||||
|
height = label:GetStringHeight()
|
||||||
|
end
|
||||||
|
|
||||||
|
-- avoid zero-height labels, since they can used as spacers
|
||||||
|
if not height or height == 0 then
|
||||||
|
height = 1
|
||||||
|
end
|
||||||
|
|
||||||
|
self.resizing = true
|
||||||
|
frame:SetHeight(height)
|
||||||
|
frame.height = height
|
||||||
|
self.resizing = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
-- set the flag to stop constant size updates
|
||||||
|
self.resizing = true
|
||||||
|
-- height is set dynamically by the text and image size
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetText()
|
||||||
|
self:SetImage(nil)
|
||||||
|
self:SetImageSize(16, 16)
|
||||||
|
self:SetColor()
|
||||||
|
self:SetFontObject()
|
||||||
|
self:SetJustifyH("LEFT")
|
||||||
|
self:SetJustifyV("TOP")
|
||||||
|
|
||||||
|
-- reset the flag
|
||||||
|
self.resizing = nil
|
||||||
|
-- run the update explicitly
|
||||||
|
UpdateImageAnchor(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["OnWidthSet"] = function(self, width)
|
||||||
|
UpdateImageAnchor(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetText"] = function(self, text)
|
||||||
|
self.label:SetText(text)
|
||||||
|
UpdateImageAnchor(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetColor"] = function(self, r, g, b)
|
||||||
|
if not (r and g and b) then
|
||||||
|
r, g, b = 1, 1, 1
|
||||||
|
end
|
||||||
|
self.label:SetVertexColor(r, g, b)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetImage"] = function(self, path, ...)
|
||||||
|
local image = self.image
|
||||||
|
image:SetTexture(path)
|
||||||
|
|
||||||
|
if image:GetTexture() then
|
||||||
|
self.imageshown = true
|
||||||
|
local n = select("#", ...)
|
||||||
|
if n == 4 or n == 8 then
|
||||||
|
image:SetTexCoord(...)
|
||||||
|
else
|
||||||
|
image:SetTexCoord(0, 1, 0, 1)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
self.imageshown = nil
|
||||||
|
end
|
||||||
|
UpdateImageAnchor(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetFont"] = function(self, font, height, flags)
|
||||||
|
self.label:SetFont(font, height, flags)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetFontObject"] = function(self, font)
|
||||||
|
self:SetFont((font or GameFontHighlightSmall):GetFont())
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetImageSize"] = function(self, width, height)
|
||||||
|
self.image:SetWidth(width)
|
||||||
|
self.image:SetHeight(height)
|
||||||
|
UpdateImageAnchor(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetJustifyH"] = function(self, justifyH)
|
||||||
|
self.label:SetJustifyH(justifyH)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetJustifyV"] = function(self, justifyV)
|
||||||
|
self.label:SetJustifyV(justifyV)
|
||||||
|
end,
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "BACKGROUND", "GameFontHighlightSmall")
|
||||||
|
local image = frame:CreateTexture(nil, "BACKGROUND")
|
||||||
|
|
||||||
|
-- create widget
|
||||||
|
local widget = {
|
||||||
|
label = label,
|
||||||
|
image = image,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,366 @@
|
|||||||
|
local Type, Version = "MultiLineEditBox", 28
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs = pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local GetCursorInfo, GetSpellInfo, ClearCursor = GetCursorInfo, GetSpellInfo, ClearCursor
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
local _G = _G
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: ACCEPT, ChatFontNormal
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
|
||||||
|
if not AceGUIMultiLineEditBoxInsertLink then
|
||||||
|
-- upgradeable hook
|
||||||
|
hooksecurefunc("ChatEdit_InsertLink", function(...) return _G.AceGUIMultiLineEditBoxInsertLink(...) end)
|
||||||
|
end
|
||||||
|
|
||||||
|
function _G.AceGUIMultiLineEditBoxInsertLink(text)
|
||||||
|
for i = 1, AceGUI:GetWidgetCount(Type) do
|
||||||
|
local editbox = _G[("MultiLineEditBox%uEdit"):format(i)]
|
||||||
|
if editbox and editbox:IsVisible() and editbox:HasFocus() then
|
||||||
|
editbox:Insert(text)
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
local function Layout(self)
|
||||||
|
self:SetHeight(self.numlines * 14 + (self.disablebutton and 19 or 41) + self.labelHeight)
|
||||||
|
|
||||||
|
if self.labelHeight == 0 then
|
||||||
|
self.scrollBar:SetPoint("TOP", self.frame, "TOP", 0, -23)
|
||||||
|
else
|
||||||
|
self.scrollBar:SetPoint("TOP", self.label, "BOTTOM", 0, -19)
|
||||||
|
end
|
||||||
|
|
||||||
|
if self.disablebutton then
|
||||||
|
self.scrollBar:SetPoint("BOTTOM", self.frame, "BOTTOM", 0, 21)
|
||||||
|
self.scrollBG:SetPoint("BOTTOMLEFT", 0, 4)
|
||||||
|
else
|
||||||
|
self.scrollBar:SetPoint("BOTTOM", self.button, "TOP", 0, 18)
|
||||||
|
self.scrollBG:SetPoint("BOTTOMLEFT", self.button, "TOPLEFT")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function OnClick(self) -- Button
|
||||||
|
self = self.obj
|
||||||
|
self.editBox:ClearFocus()
|
||||||
|
if not self:Fire("OnEnterPressed", self.editBox:GetText()) then
|
||||||
|
self.button:Disable()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnCursorChanged(self, _, y, _, cursorHeight) -- EditBox
|
||||||
|
self, y = self.obj.scrollFrame, -y
|
||||||
|
local offset = self:GetVerticalScroll()
|
||||||
|
if y < offset then
|
||||||
|
self:SetVerticalScroll(y)
|
||||||
|
else
|
||||||
|
y = y + cursorHeight - self:GetHeight()
|
||||||
|
if y > offset then
|
||||||
|
self:SetVerticalScroll(y)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnEditFocusLost(self) -- EditBox
|
||||||
|
self:HighlightText(0, 0)
|
||||||
|
self.obj:Fire("OnEditFocusLost")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnEnter(self) -- EditBox / ScrollFrame
|
||||||
|
self = self.obj
|
||||||
|
if not self.entered then
|
||||||
|
self.entered = true
|
||||||
|
self:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnLeave(self) -- EditBox / ScrollFrame
|
||||||
|
self = self.obj
|
||||||
|
if self.entered then
|
||||||
|
self.entered = nil
|
||||||
|
self:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnMouseUp(self) -- ScrollFrame
|
||||||
|
self = self.obj.editBox
|
||||||
|
self:SetFocus()
|
||||||
|
self:SetCursorPosition(self:GetNumLetters())
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnReceiveDrag(self) -- EditBox / ScrollFrame
|
||||||
|
local type, id, info = GetCursorInfo()
|
||||||
|
if type == "spell" then
|
||||||
|
info = GetSpellInfo(id, info)
|
||||||
|
elseif type ~= "item" then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
ClearCursor()
|
||||||
|
self = self.obj
|
||||||
|
local editBox = self.editBox
|
||||||
|
if not editBox:HasFocus() then
|
||||||
|
editBox:SetFocus()
|
||||||
|
editBox:SetCursorPosition(editBox:GetNumLetters())
|
||||||
|
end
|
||||||
|
editBox:Insert(info)
|
||||||
|
self.button:Enable()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnSizeChanged(self, width, height) -- ScrollFrame
|
||||||
|
self.obj.editBox:SetWidth(width)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnTextChanged(self, userInput) -- EditBox
|
||||||
|
if userInput then
|
||||||
|
self = self.obj
|
||||||
|
self:Fire("OnTextChanged", self.editBox:GetText())
|
||||||
|
self.button:Enable()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnTextSet(self) -- EditBox
|
||||||
|
self:HighlightText(0, 0)
|
||||||
|
self:SetCursorPosition(self:GetNumLetters())
|
||||||
|
self:SetCursorPosition(0)
|
||||||
|
self.obj.button:Disable()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnVerticalScroll(self, offset) -- ScrollFrame
|
||||||
|
local editBox = self.obj.editBox
|
||||||
|
editBox:SetHitRectInsets(0, 0, offset, editBox:GetHeight() - offset - self:GetHeight())
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnShowFocus(frame)
|
||||||
|
frame.obj.editBox:SetFocus()
|
||||||
|
frame:SetScript("OnShow", nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function OnEditFocusGained(frame)
|
||||||
|
AceGUI:SetFocus(frame.obj)
|
||||||
|
frame.obj:Fire("OnEditFocusGained")
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self.editBox:SetText("")
|
||||||
|
self:SetDisabled(false)
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:DisableButton(false)
|
||||||
|
self:SetNumLines()
|
||||||
|
self.entered = nil
|
||||||
|
self:SetMaxLetters(0)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["OnRelease"] = function(self)
|
||||||
|
self:ClearFocus()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
local editBox = self.editBox
|
||||||
|
if disabled then
|
||||||
|
editBox:ClearFocus()
|
||||||
|
editBox:EnableMouse(false)
|
||||||
|
editBox:SetTextColor(0.5, 0.5, 0.5)
|
||||||
|
self.label:SetTextColor(0.5, 0.5, 0.5)
|
||||||
|
self.scrollFrame:EnableMouse(false)
|
||||||
|
self.button:Disable()
|
||||||
|
else
|
||||||
|
editBox:EnableMouse(true)
|
||||||
|
editBox:SetTextColor(1, 1, 1)
|
||||||
|
self.label:SetTextColor(1, 0.82, 0)
|
||||||
|
self.scrollFrame:EnableMouse(true)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetLabel"] = function(self, text)
|
||||||
|
if text and text ~= "" then
|
||||||
|
self.label:SetText(text)
|
||||||
|
if self.labelHeight ~= 10 then
|
||||||
|
self.labelHeight = 10
|
||||||
|
self.label:Show()
|
||||||
|
end
|
||||||
|
elseif self.labelHeight ~= 0 then
|
||||||
|
self.labelHeight = 0
|
||||||
|
self.label:Hide()
|
||||||
|
end
|
||||||
|
Layout(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetNumLines"] = function(self, value)
|
||||||
|
if not value or value < 4 then
|
||||||
|
value = 4
|
||||||
|
end
|
||||||
|
self.numlines = value
|
||||||
|
Layout(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetText"] = function(self, text)
|
||||||
|
self.editBox:SetText(text)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["GetText"] = function(self)
|
||||||
|
return self.editBox:GetText()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetMaxLetters"] = function (self, num)
|
||||||
|
self.editBox:SetMaxLetters(num or 0)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["DisableButton"] = function(self, disabled)
|
||||||
|
self.disablebutton = disabled
|
||||||
|
if disabled then
|
||||||
|
self.button:Hide()
|
||||||
|
else
|
||||||
|
self.button:Show()
|
||||||
|
end
|
||||||
|
Layout(self)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["ClearFocus"] = function(self)
|
||||||
|
self.editBox:ClearFocus()
|
||||||
|
self.frame:SetScript("OnShow", nil)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetFocus"] = function(self)
|
||||||
|
self.editBox:SetFocus()
|
||||||
|
if not self.frame:IsShown() then
|
||||||
|
self.frame:SetScript("OnShow", OnShowFocus)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["HighlightText"] = function(self, from, to)
|
||||||
|
self.editBox:HighlightText(from, to)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["GetCursorPosition"] = function(self)
|
||||||
|
return self.editBox:GetCursorPosition()
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetCursorPosition"] = function(self, ...)
|
||||||
|
return self.editBox:SetCursorPosition(...)
|
||||||
|
end,
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local backdrop = {
|
||||||
|
bgFile = [[Interface\Tooltips\UI-Tooltip-Background]],
|
||||||
|
edgeFile = [[Interface\Tooltips\UI-Tooltip-Border]], edgeSize = 16,
|
||||||
|
insets = { left = 4, right = 3, top = 4, bottom = 3 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
frame:Hide()
|
||||||
|
|
||||||
|
local widgetNum = AceGUI:GetNextWidgetNum(Type)
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "OVERLAY", "GameFontNormalSmall")
|
||||||
|
label:SetPoint("TOPLEFT", frame, "TOPLEFT", 0, -4)
|
||||||
|
label:SetPoint("TOPRIGHT", frame, "TOPRIGHT", 0, -4)
|
||||||
|
label:SetJustifyH("LEFT")
|
||||||
|
label:SetText(ACCEPT)
|
||||||
|
label:SetHeight(10)
|
||||||
|
|
||||||
|
local button = CreateFrame("Button", ("%s%dButton"):format(Type, widgetNum), frame, "UIPanelButtonTemplate")
|
||||||
|
button:SetPoint("BOTTOMLEFT", 0, 4)
|
||||||
|
button:SetHeight(22)
|
||||||
|
button:SetWidth(label:GetStringWidth() + 24)
|
||||||
|
button:SetText(ACCEPT)
|
||||||
|
button:SetScript("OnClick", OnClick)
|
||||||
|
button:Disable()
|
||||||
|
|
||||||
|
local text = button:GetFontString()
|
||||||
|
text:ClearAllPoints()
|
||||||
|
text:SetPoint("TOPLEFT", button, "TOPLEFT", 5, -5)
|
||||||
|
text:SetPoint("BOTTOMRIGHT", button, "BOTTOMRIGHT", -5, 1)
|
||||||
|
text:SetJustifyV("MIDDLE")
|
||||||
|
|
||||||
|
local scrollBG = CreateFrame("Frame", nil, frame)
|
||||||
|
scrollBG:SetBackdrop(backdrop)
|
||||||
|
scrollBG:SetBackdropColor(0, 0, 0)
|
||||||
|
scrollBG:SetBackdropBorderColor(0.4, 0.4, 0.4)
|
||||||
|
|
||||||
|
local scrollFrame = CreateFrame("ScrollFrame", ("%s%dScrollFrame"):format(Type, widgetNum), frame, "UIPanelScrollFrameTemplate")
|
||||||
|
|
||||||
|
local scrollBar = _G[scrollFrame:GetName() .. "ScrollBar"]
|
||||||
|
scrollBar:ClearAllPoints()
|
||||||
|
scrollBar:SetPoint("TOP", label, "BOTTOM", 0, -19)
|
||||||
|
scrollBar:SetPoint("BOTTOM", button, "TOP", 0, 18)
|
||||||
|
scrollBar:SetPoint("RIGHT", frame, "RIGHT")
|
||||||
|
|
||||||
|
scrollBG:SetPoint("TOPRIGHT", scrollBar, "TOPLEFT", 0, 19)
|
||||||
|
scrollBG:SetPoint("BOTTOMLEFT", button, "TOPLEFT")
|
||||||
|
|
||||||
|
scrollFrame:SetPoint("TOPLEFT", scrollBG, "TOPLEFT", 5, -6)
|
||||||
|
scrollFrame:SetPoint("BOTTOMRIGHT", scrollBG, "BOTTOMRIGHT", -4, 4)
|
||||||
|
scrollFrame:SetScript("OnEnter", OnEnter)
|
||||||
|
scrollFrame:SetScript("OnLeave", OnLeave)
|
||||||
|
scrollFrame:SetScript("OnMouseUp", OnMouseUp)
|
||||||
|
scrollFrame:SetScript("OnReceiveDrag", OnReceiveDrag)
|
||||||
|
scrollFrame:SetScript("OnSizeChanged", OnSizeChanged)
|
||||||
|
scrollFrame:HookScript("OnVerticalScroll", OnVerticalScroll)
|
||||||
|
|
||||||
|
local editBox = CreateFrame("EditBox", ("%s%dEdit"):format(Type, widgetNum), scrollFrame)
|
||||||
|
editBox:SetAllPoints()
|
||||||
|
editBox:SetFontObject(ChatFontNormal)
|
||||||
|
editBox:SetMultiLine(true)
|
||||||
|
editBox:EnableMouse(true)
|
||||||
|
editBox:SetAutoFocus(false)
|
||||||
|
editBox:SetCountInvisibleLetters(false)
|
||||||
|
editBox:SetScript("OnCursorChanged", OnCursorChanged)
|
||||||
|
editBox:SetScript("OnEditFocusLost", OnEditFocusLost)
|
||||||
|
editBox:SetScript("OnEnter", OnEnter)
|
||||||
|
editBox:SetScript("OnEscapePressed", editBox.ClearFocus)
|
||||||
|
editBox:SetScript("OnLeave", OnLeave)
|
||||||
|
editBox:SetScript("OnMouseDown", OnReceiveDrag)
|
||||||
|
editBox:SetScript("OnReceiveDrag", OnReceiveDrag)
|
||||||
|
editBox:SetScript("OnTextChanged", OnTextChanged)
|
||||||
|
editBox:SetScript("OnTextSet", OnTextSet)
|
||||||
|
editBox:SetScript("OnEditFocusGained", OnEditFocusGained)
|
||||||
|
|
||||||
|
|
||||||
|
scrollFrame:SetScrollChild(editBox)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
button = button,
|
||||||
|
editBox = editBox,
|
||||||
|
frame = frame,
|
||||||
|
label = label,
|
||||||
|
labelHeight = 10,
|
||||||
|
numlines = 4,
|
||||||
|
scrollBar = scrollBar,
|
||||||
|
scrollBG = scrollBG,
|
||||||
|
scrollFrame = scrollFrame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
button.obj, editBox.obj, scrollFrame.obj = widget, widget, widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type, Constructor, Version)
|
@ -0,0 +1,284 @@
|
|||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Slider Widget
|
||||||
|
Graphical Slider, like, for Range values.
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local Type, Version = "Slider", 22
|
||||||
|
local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
|
||||||
|
if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local min, max, floor = math.min, math.max, math.floor
|
||||||
|
local tonumber, pairs = tonumber, pairs
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local PlaySound = PlaySound
|
||||||
|
local CreateFrame, UIParent = CreateFrame, UIParent
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: GameFontHighlightSmall
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Support functions
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function UpdateText(self)
|
||||||
|
local value = self.value or 0
|
||||||
|
if self.ispercent then
|
||||||
|
self.editbox:SetText(("%s%%"):format(floor(value * 1000 + 0.5) / 10))
|
||||||
|
else
|
||||||
|
self.editbox:SetText(floor(value * 100 + 0.5) / 100)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function UpdateLabels(self)
|
||||||
|
local min, max = (self.min or 0), (self.max or 100)
|
||||||
|
if self.ispercent then
|
||||||
|
self.lowtext:SetFormattedText("%s%%", (min * 100))
|
||||||
|
self.hightext:SetFormattedText("%s%%", (max * 100))
|
||||||
|
else
|
||||||
|
self.lowtext:SetText(min)
|
||||||
|
self.hightext:SetText(max)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Scripts
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local function Control_OnEnter(frame)
|
||||||
|
frame.obj:Fire("OnEnter")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Control_OnLeave(frame)
|
||||||
|
frame.obj:Fire("OnLeave")
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Frame_OnMouseDown(frame)
|
||||||
|
frame.obj.slider:EnableMouseWheel(true)
|
||||||
|
AceGUI:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Slider_OnValueChanged(frame, newvalue)
|
||||||
|
local self = frame.obj
|
||||||
|
if not frame.setup then
|
||||||
|
if self.step and self.step > 0 then
|
||||||
|
local min_value = self.min or 0
|
||||||
|
newvalue = floor((newvalue - min_value) / self.step + 0.5) * self.step + min_value
|
||||||
|
end
|
||||||
|
if newvalue ~= self.value and not self.disabled then
|
||||||
|
self.value = newvalue
|
||||||
|
self:Fire("OnValueChanged", newvalue)
|
||||||
|
end
|
||||||
|
if self.value then
|
||||||
|
UpdateText(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Slider_OnMouseUp(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
self:Fire("OnMouseUp", self.value)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Slider_OnMouseWheel(frame, v)
|
||||||
|
local self = frame.obj
|
||||||
|
if not self.disabled then
|
||||||
|
local value = self.value
|
||||||
|
if v > 0 then
|
||||||
|
value = min(value + (self.step or 1), self.max)
|
||||||
|
else
|
||||||
|
value = max(value - (self.step or 1), self.min)
|
||||||
|
end
|
||||||
|
self.slider:SetValue(value)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnEscapePressed(frame)
|
||||||
|
frame:ClearFocus()
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnEnterPressed(frame)
|
||||||
|
local self = frame.obj
|
||||||
|
local value = frame:GetText()
|
||||||
|
if self.ispercent then
|
||||||
|
value = value:gsub('%%', '')
|
||||||
|
value = tonumber(value) / 100
|
||||||
|
else
|
||||||
|
value = tonumber(value)
|
||||||
|
end
|
||||||
|
|
||||||
|
if value then
|
||||||
|
PlaySound(856) -- SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON
|
||||||
|
self.slider:SetValue(value)
|
||||||
|
self:Fire("OnMouseUp", value)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnEnter(frame)
|
||||||
|
frame:SetBackdropBorderColor(0.5, 0.5, 0.5, 1)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function EditBox_OnLeave(frame)
|
||||||
|
frame:SetBackdropBorderColor(0.3, 0.3, 0.3, 0.8)
|
||||||
|
end
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Methods
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local methods = {
|
||||||
|
["OnAcquire"] = function(self)
|
||||||
|
self:SetWidth(200)
|
||||||
|
self:SetHeight(44)
|
||||||
|
self:SetDisabled(false)
|
||||||
|
self:SetIsPercent(nil)
|
||||||
|
self:SetSliderValues(0,100,1)
|
||||||
|
self:SetValue(0)
|
||||||
|
self.slider:EnableMouseWheel(false)
|
||||||
|
end,
|
||||||
|
|
||||||
|
-- ["OnRelease"] = nil,
|
||||||
|
|
||||||
|
["SetDisabled"] = function(self, disabled)
|
||||||
|
self.disabled = disabled
|
||||||
|
if disabled then
|
||||||
|
self.slider:EnableMouse(false)
|
||||||
|
self.label:SetTextColor(.5, .5, .5)
|
||||||
|
self.hightext:SetTextColor(.5, .5, .5)
|
||||||
|
self.lowtext:SetTextColor(.5, .5, .5)
|
||||||
|
--self.valuetext:SetTextColor(.5, .5, .5)
|
||||||
|
self.editbox:SetTextColor(.5, .5, .5)
|
||||||
|
self.editbox:EnableMouse(false)
|
||||||
|
self.editbox:ClearFocus()
|
||||||
|
else
|
||||||
|
self.slider:EnableMouse(true)
|
||||||
|
self.label:SetTextColor(1, .82, 0)
|
||||||
|
self.hightext:SetTextColor(1, 1, 1)
|
||||||
|
self.lowtext:SetTextColor(1, 1, 1)
|
||||||
|
--self.valuetext:SetTextColor(1, 1, 1)
|
||||||
|
self.editbox:SetTextColor(1, 1, 1)
|
||||||
|
self.editbox:EnableMouse(true)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetValue"] = function(self, value)
|
||||||
|
self.slider.setup = true
|
||||||
|
self.slider:SetValue(value)
|
||||||
|
self.value = value
|
||||||
|
UpdateText(self)
|
||||||
|
self.slider.setup = nil
|
||||||
|
end,
|
||||||
|
|
||||||
|
["GetValue"] = function(self)
|
||||||
|
return self.value
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetLabel"] = function(self, text)
|
||||||
|
self.label:SetText(text)
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetSliderValues"] = function(self, min, max, step)
|
||||||
|
local frame = self.slider
|
||||||
|
frame.setup = true
|
||||||
|
self.min = min
|
||||||
|
self.max = max
|
||||||
|
self.step = step
|
||||||
|
frame:SetMinMaxValues(min or 0,max or 100)
|
||||||
|
UpdateLabels(self)
|
||||||
|
frame:SetValueStep(step or 1)
|
||||||
|
if self.value then
|
||||||
|
frame:SetValue(self.value)
|
||||||
|
end
|
||||||
|
frame.setup = nil
|
||||||
|
end,
|
||||||
|
|
||||||
|
["SetIsPercent"] = function(self, value)
|
||||||
|
self.ispercent = value
|
||||||
|
UpdateLabels(self)
|
||||||
|
UpdateText(self)
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
--[[-----------------------------------------------------------------------------
|
||||||
|
Constructor
|
||||||
|
-------------------------------------------------------------------------------]]
|
||||||
|
local SliderBackdrop = {
|
||||||
|
bgFile = "Interface\\Buttons\\UI-SliderBar-Background",
|
||||||
|
edgeFile = "Interface\\Buttons\\UI-SliderBar-Border",
|
||||||
|
tile = true, tileSize = 8, edgeSize = 8,
|
||||||
|
insets = { left = 3, right = 3, top = 6, bottom = 6 }
|
||||||
|
}
|
||||||
|
|
||||||
|
local ManualBackdrop = {
|
||||||
|
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
edgeFile = "Interface\\ChatFrame\\ChatFrameBackground",
|
||||||
|
tile = true, edgeSize = 1, tileSize = 5,
|
||||||
|
}
|
||||||
|
|
||||||
|
local function Constructor()
|
||||||
|
local frame = CreateFrame("Frame", nil, UIParent)
|
||||||
|
|
||||||
|
frame:EnableMouse(true)
|
||||||
|
frame:SetScript("OnMouseDown", Frame_OnMouseDown)
|
||||||
|
|
||||||
|
local label = frame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||||
|
label:SetPoint("TOPLEFT")
|
||||||
|
label:SetPoint("TOPRIGHT")
|
||||||
|
label:SetJustifyH("CENTER")
|
||||||
|
label:SetHeight(15)
|
||||||
|
|
||||||
|
local slider = CreateFrame("Slider", nil, frame)
|
||||||
|
slider:SetOrientation("HORIZONTAL")
|
||||||
|
slider:SetHeight(15)
|
||||||
|
slider:SetHitRectInsets(0, 0, -10, 0)
|
||||||
|
slider:SetBackdrop(SliderBackdrop)
|
||||||
|
slider:SetThumbTexture("Interface\\Buttons\\UI-SliderBar-Button-Horizontal")
|
||||||
|
slider:SetPoint("TOP", label, "BOTTOM")
|
||||||
|
slider:SetPoint("LEFT", 3, 0)
|
||||||
|
slider:SetPoint("RIGHT", -3, 0)
|
||||||
|
slider:SetValue(0)
|
||||||
|
slider:SetScript("OnValueChanged",Slider_OnValueChanged)
|
||||||
|
slider:SetScript("OnEnter", Control_OnEnter)
|
||||||
|
slider:SetScript("OnLeave", Control_OnLeave)
|
||||||
|
slider:SetScript("OnMouseUp", Slider_OnMouseUp)
|
||||||
|
slider:SetScript("OnMouseWheel", Slider_OnMouseWheel)
|
||||||
|
|
||||||
|
local lowtext = slider:CreateFontString(nil, "ARTWORK", "GameFontHighlightSmall")
|
||||||
|
lowtext:SetPoint("TOPLEFT", slider, "BOTTOMLEFT", 2, 3)
|
||||||
|
|
||||||
|
local hightext = slider:CreateFontString(nil, "ARTWORK", "GameFontHighlightSmall")
|
||||||
|
hightext:SetPoint("TOPRIGHT", slider, "BOTTOMRIGHT", -2, 3)
|
||||||
|
|
||||||
|
local editbox = CreateFrame("EditBox", nil, frame)
|
||||||
|
editbox:SetAutoFocus(false)
|
||||||
|
editbox:SetFontObject(GameFontHighlightSmall)
|
||||||
|
editbox:SetPoint("TOP", slider, "BOTTOM")
|
||||||
|
editbox:SetHeight(14)
|
||||||
|
editbox:SetWidth(70)
|
||||||
|
editbox:SetJustifyH("CENTER")
|
||||||
|
editbox:EnableMouse(true)
|
||||||
|
editbox:SetBackdrop(ManualBackdrop)
|
||||||
|
editbox:SetBackdropColor(0, 0, 0, 0.5)
|
||||||
|
editbox:SetBackdropBorderColor(0.3, 0.3, 0.30, 0.80)
|
||||||
|
editbox:SetScript("OnEnter", EditBox_OnEnter)
|
||||||
|
editbox:SetScript("OnLeave", EditBox_OnLeave)
|
||||||
|
editbox:SetScript("OnEnterPressed", EditBox_OnEnterPressed)
|
||||||
|
editbox:SetScript("OnEscapePressed", EditBox_OnEscapePressed)
|
||||||
|
|
||||||
|
local widget = {
|
||||||
|
label = label,
|
||||||
|
slider = slider,
|
||||||
|
lowtext = lowtext,
|
||||||
|
hightext = hightext,
|
||||||
|
editbox = editbox,
|
||||||
|
alignoffset = 25,
|
||||||
|
frame = frame,
|
||||||
|
type = Type
|
||||||
|
}
|
||||||
|
for method, func in pairs(methods) do
|
||||||
|
widget[method] = func
|
||||||
|
end
|
||||||
|
slider.obj, editbox.obj = widget, widget
|
||||||
|
|
||||||
|
return AceGUI:RegisterAsWidget(widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
AceGUI:RegisterWidgetType(Type,Constructor,Version)
|
@ -0,0 +1,511 @@
|
|||||||
|
--- **AceHook-3.0** offers safe Hooking/Unhooking of functions, methods and frame scripts.
|
||||||
|
-- Using AceHook-3.0 is recommended when you need to unhook your hooks again, so the hook chain isn't broken
|
||||||
|
-- when you manually restore the original function.
|
||||||
|
--
|
||||||
|
-- **AceHook-3.0** can be embeded into your addon, either explicitly by calling AceHook:Embed(MyAddon) or by
|
||||||
|
-- specifying it as an embeded library in your AceAddon. All functions will be available on your addon object
|
||||||
|
-- and can be accessed directly, without having to explicitly call AceHook itself.\\
|
||||||
|
-- It is recommended to embed AceHook, otherwise you'll have to specify a custom `self` on all calls you
|
||||||
|
-- make into AceHook.
|
||||||
|
-- @class file
|
||||||
|
-- @name AceHook-3.0
|
||||||
|
-- @release $Id: AceHook-3.0.lua 1202 2019-05-15 23:11:22Z nevcairiel $
|
||||||
|
local ACEHOOK_MAJOR, ACEHOOK_MINOR = "AceHook-3.0", 8
|
||||||
|
local AceHook, oldminor = LibStub:NewLibrary(ACEHOOK_MAJOR, ACEHOOK_MINOR)
|
||||||
|
|
||||||
|
if not AceHook then return end -- No upgrade needed
|
||||||
|
|
||||||
|
AceHook.embeded = AceHook.embeded or {}
|
||||||
|
AceHook.registry = AceHook.registry or setmetatable({}, {__index = function(tbl, key) tbl[key] = {} return tbl[key] end })
|
||||||
|
AceHook.handlers = AceHook.handlers or {}
|
||||||
|
AceHook.actives = AceHook.actives or {}
|
||||||
|
AceHook.scripts = AceHook.scripts or {}
|
||||||
|
AceHook.onceSecure = AceHook.onceSecure or {}
|
||||||
|
AceHook.hooks = AceHook.hooks or {}
|
||||||
|
|
||||||
|
-- local upvalues
|
||||||
|
local registry = AceHook.registry
|
||||||
|
local handlers = AceHook.handlers
|
||||||
|
local actives = AceHook.actives
|
||||||
|
local scripts = AceHook.scripts
|
||||||
|
local onceSecure = AceHook.onceSecure
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local pairs, next, type = pairs, next, type
|
||||||
|
local format = string.format
|
||||||
|
local assert, error = assert, error
|
||||||
|
|
||||||
|
-- WoW APIs
|
||||||
|
local issecurevariable, hooksecurefunc = issecurevariable, hooksecurefunc
|
||||||
|
local _G = _G
|
||||||
|
|
||||||
|
-- functions for later definition
|
||||||
|
local donothing, createHook, hook
|
||||||
|
|
||||||
|
local protectedScripts = {
|
||||||
|
OnClick = true,
|
||||||
|
}
|
||||||
|
|
||||||
|
-- upgrading of embeded is done at the bottom of the file
|
||||||
|
|
||||||
|
local mixins = {
|
||||||
|
"Hook", "SecureHook",
|
||||||
|
"HookScript", "SecureHookScript",
|
||||||
|
"Unhook", "UnhookAll",
|
||||||
|
"IsHooked",
|
||||||
|
"RawHook", "RawHookScript"
|
||||||
|
}
|
||||||
|
|
||||||
|
-- AceHook:Embed( target )
|
||||||
|
-- target (object) - target object to embed AceHook in
|
||||||
|
--
|
||||||
|
-- Embeds AceEevent into the target object making the functions from the mixins list available on target:..
|
||||||
|
function AceHook:Embed( target )
|
||||||
|
for k, v in pairs( mixins ) do
|
||||||
|
target[v] = self[v]
|
||||||
|
end
|
||||||
|
self.embeded[target] = true
|
||||||
|
-- inject the hooks table safely
|
||||||
|
target.hooks = target.hooks or {}
|
||||||
|
return target
|
||||||
|
end
|
||||||
|
|
||||||
|
-- AceHook:OnEmbedDisable( target )
|
||||||
|
-- target (object) - target object that is being disabled
|
||||||
|
--
|
||||||
|
-- Unhooks all hooks when the target disables.
|
||||||
|
-- this method should be called by the target manually or by an addon framework
|
||||||
|
function AceHook:OnEmbedDisable( target )
|
||||||
|
target:UnhookAll()
|
||||||
|
end
|
||||||
|
|
||||||
|
function createHook(self, handler, orig, secure, failsafe)
|
||||||
|
local uid
|
||||||
|
local method = type(handler) == "string"
|
||||||
|
if failsafe and not secure then
|
||||||
|
-- failsafe hook creation
|
||||||
|
uid = function(...)
|
||||||
|
if actives[uid] then
|
||||||
|
if method then
|
||||||
|
self[handler](self, ...)
|
||||||
|
else
|
||||||
|
handler(...)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return orig(...)
|
||||||
|
end
|
||||||
|
-- /failsafe hook
|
||||||
|
else
|
||||||
|
-- all other hooks
|
||||||
|
uid = function(...)
|
||||||
|
if actives[uid] then
|
||||||
|
if method then
|
||||||
|
return self[handler](self, ...)
|
||||||
|
else
|
||||||
|
return handler(...)
|
||||||
|
end
|
||||||
|
elseif not secure then -- backup on non secure
|
||||||
|
return orig(...)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
-- /hook
|
||||||
|
end
|
||||||
|
return uid
|
||||||
|
end
|
||||||
|
|
||||||
|
function donothing() end
|
||||||
|
|
||||||
|
function hook(self, obj, method, handler, script, secure, raw, forceSecure, usage)
|
||||||
|
if not handler then handler = method end
|
||||||
|
|
||||||
|
-- These asserts make sure AceHooks's devs play by the rules.
|
||||||
|
assert(not script or type(script) == "boolean")
|
||||||
|
assert(not secure or type(secure) == "boolean")
|
||||||
|
assert(not raw or type(raw) == "boolean")
|
||||||
|
assert(not forceSecure or type(forceSecure) == "boolean")
|
||||||
|
assert(usage)
|
||||||
|
|
||||||
|
-- Error checking Battery!
|
||||||
|
if obj and type(obj) ~= "table" then
|
||||||
|
error(format("%s: 'object' - nil or table expected got %s", usage, type(obj)), 3)
|
||||||
|
end
|
||||||
|
if type(method) ~= "string" then
|
||||||
|
error(format("%s: 'method' - string expected got %s", usage, type(method)), 3)
|
||||||
|
end
|
||||||
|
if type(handler) ~= "string" and type(handler) ~= "function" then
|
||||||
|
error(format("%s: 'handler' - nil, string, or function expected got %s", usage, type(handler)), 3)
|
||||||
|
end
|
||||||
|
if type(handler) == "string" and type(self[handler]) ~= "function" then
|
||||||
|
error(format("%s: 'handler' - Handler specified does not exist at self[handler]", usage), 3)
|
||||||
|
end
|
||||||
|
if script then
|
||||||
|
if not obj or not obj.GetScript or not obj:HasScript(method) then
|
||||||
|
error(format("%s: You can only hook a script on a frame object", usage), 3)
|
||||||
|
end
|
||||||
|
if not secure and obj.IsProtected and obj:IsProtected() and protectedScripts[method] then
|
||||||
|
error(format("Cannot hook secure script %q; Use SecureHookScript(obj, method, [handler]) instead.", method), 3)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
local issecure
|
||||||
|
if obj then
|
||||||
|
issecure = onceSecure[obj] and onceSecure[obj][method] or issecurevariable(obj, method)
|
||||||
|
else
|
||||||
|
issecure = onceSecure[method] or issecurevariable(method)
|
||||||
|
end
|
||||||
|
if issecure then
|
||||||
|
if forceSecure then
|
||||||
|
if obj then
|
||||||
|
onceSecure[obj] = onceSecure[obj] or {}
|
||||||
|
onceSecure[obj][method] = true
|
||||||
|
else
|
||||||
|
onceSecure[method] = true
|
||||||
|
end
|
||||||
|
elseif not secure then
|
||||||
|
error(format("%s: Attempt to hook secure function %s. Use `SecureHook' or add `true' to the argument list to override.", usage, method), 3)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local uid
|
||||||
|
if obj then
|
||||||
|
uid = registry[self][obj] and registry[self][obj][method]
|
||||||
|
else
|
||||||
|
uid = registry[self][method]
|
||||||
|
end
|
||||||
|
|
||||||
|
if uid then
|
||||||
|
if actives[uid] then
|
||||||
|
-- Only two sane choices exist here. We either a) error 100% of the time or b) always unhook and then hook
|
||||||
|
-- choice b would likely lead to odd debuging conditions or other mysteries so we're going with a.
|
||||||
|
error(format("Attempting to rehook already active hook %s.", method))
|
||||||
|
end
|
||||||
|
|
||||||
|
if handlers[uid] == handler then -- turn on a decative hook, note enclosures break this ability, small memory leak
|
||||||
|
actives[uid] = true
|
||||||
|
return
|
||||||
|
elseif obj then -- is there any reason not to call unhook instead of doing the following several lines?
|
||||||
|
if self.hooks and self.hooks[obj] then
|
||||||
|
self.hooks[obj][method] = nil
|
||||||
|
end
|
||||||
|
registry[self][obj][method] = nil
|
||||||
|
else
|
||||||
|
if self.hooks then
|
||||||
|
self.hooks[method] = nil
|
||||||
|
end
|
||||||
|
registry[self][method] = nil
|
||||||
|
end
|
||||||
|
handlers[uid], actives[uid], scripts[uid] = nil, nil, nil
|
||||||
|
uid = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
local orig
|
||||||
|
if script then
|
||||||
|
orig = obj:GetScript(method) or donothing
|
||||||
|
elseif obj then
|
||||||
|
orig = obj[method]
|
||||||
|
else
|
||||||
|
orig = _G[method]
|
||||||
|
end
|
||||||
|
|
||||||
|
if not orig then
|
||||||
|
error(format("%s: Attempting to hook a non existing target", usage), 3)
|
||||||
|
end
|
||||||
|
|
||||||
|
uid = createHook(self, handler, orig, secure, not (raw or secure))
|
||||||
|
|
||||||
|
if obj then
|
||||||
|
self.hooks[obj] = self.hooks[obj] or {}
|
||||||
|
registry[self][obj] = registry[self][obj] or {}
|
||||||
|
registry[self][obj][method] = uid
|
||||||
|
|
||||||
|
if not secure then
|
||||||
|
self.hooks[obj][method] = orig
|
||||||
|
end
|
||||||
|
|
||||||
|
if script then
|
||||||
|
if not secure then
|
||||||
|
obj:SetScript(method, uid)
|
||||||
|
else
|
||||||
|
obj:HookScript(method, uid)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
if not secure then
|
||||||
|
obj[method] = uid
|
||||||
|
else
|
||||||
|
hooksecurefunc(obj, method, uid)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
registry[self][method] = uid
|
||||||
|
|
||||||
|
if not secure then
|
||||||
|
_G[method] = uid
|
||||||
|
self.hooks[method] = orig
|
||||||
|
else
|
||||||
|
hooksecurefunc(method, uid)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
actives[uid], handlers[uid], scripts[uid] = true, handler, script and true or nil
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Hook a function or a method on an object.
|
||||||
|
-- The hook created will be a "safe hook", that means that your handler will be called
|
||||||
|
-- before the hooked function ("Pre-Hook"), and you don't have to call the original function yourself,
|
||||||
|
-- however you cannot stop the execution of the function, or modify any of the arguments/return values.\\
|
||||||
|
-- This type of hook is typically used if you need to know if some function got called, and don't want to modify it.
|
||||||
|
-- @paramsig [object], method, [handler], [hookSecure]
|
||||||
|
-- @param object The object to hook a method from
|
||||||
|
-- @param method If object was specified, the name of the method, or the name of the function to hook.
|
||||||
|
-- @param handler The handler for the hook, a funcref or a method name. (Defaults to the name of the hooked function)
|
||||||
|
-- @param hookSecure If true, AceHook will allow hooking of secure functions.
|
||||||
|
-- @usage
|
||||||
|
-- -- create an addon with AceHook embeded
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("HookDemo", "AceHook-3.0")
|
||||||
|
--
|
||||||
|
-- function MyAddon:OnEnable()
|
||||||
|
-- -- Hook ActionButton_UpdateHotkeys, overwriting the secure status
|
||||||
|
-- self:Hook("ActionButton_UpdateHotkeys", true)
|
||||||
|
-- end
|
||||||
|
--
|
||||||
|
-- function MyAddon:ActionButton_UpdateHotkeys(button, type)
|
||||||
|
-- print(button:GetName() .. " is updating its HotKey")
|
||||||
|
-- end
|
||||||
|
function AceHook:Hook(object, method, handler, hookSecure)
|
||||||
|
if type(object) == "string" then
|
||||||
|
method, handler, hookSecure, object = object, method, handler, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
if handler == true then
|
||||||
|
handler, hookSecure = nil, true
|
||||||
|
end
|
||||||
|
|
||||||
|
hook(self, object, method, handler, false, false, false, hookSecure or false, "Usage: Hook([object], method, [handler], [hookSecure])")
|
||||||
|
end
|
||||||
|
|
||||||
|
--- RawHook a function or a method on an object.
|
||||||
|
-- The hook created will be a "raw hook", that means that your handler will completly replace
|
||||||
|
-- the original function, and your handler has to call the original function (or not, depending on your intentions).\\
|
||||||
|
-- The original function will be stored in `self.hooks[object][method]` or `self.hooks[functionName]` respectively.\\
|
||||||
|
-- This type of hook can be used for all purposes, and is usually the most common case when you need to modify arguments
|
||||||
|
-- or want to control execution of the original function.
|
||||||
|
-- @paramsig [object], method, [handler], [hookSecure]
|
||||||
|
-- @param object The object to hook a method from
|
||||||
|
-- @param method If object was specified, the name of the method, or the name of the function to hook.
|
||||||
|
-- @param handler The handler for the hook, a funcref or a method name. (Defaults to the name of the hooked function)
|
||||||
|
-- @param hookSecure If true, AceHook will allow hooking of secure functions.
|
||||||
|
-- @usage
|
||||||
|
-- -- create an addon with AceHook embeded
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("HookDemo", "AceHook-3.0")
|
||||||
|
--
|
||||||
|
-- function MyAddon:OnEnable()
|
||||||
|
-- -- Hook ActionButton_UpdateHotkeys, overwriting the secure status
|
||||||
|
-- self:RawHook("ActionButton_UpdateHotkeys", true)
|
||||||
|
-- end
|
||||||
|
--
|
||||||
|
-- function MyAddon:ActionButton_UpdateHotkeys(button, type)
|
||||||
|
-- if button:GetName() == "MyButton" then
|
||||||
|
-- -- do stuff here
|
||||||
|
-- else
|
||||||
|
-- self.hooks.ActionButton_UpdateHotkeys(button, type)
|
||||||
|
-- end
|
||||||
|
-- end
|
||||||
|
function AceHook:RawHook(object, method, handler, hookSecure)
|
||||||
|
if type(object) == "string" then
|
||||||
|
method, handler, hookSecure, object = object, method, handler, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
if handler == true then
|
||||||
|
handler, hookSecure = nil, true
|
||||||
|
end
|
||||||
|
|
||||||
|
hook(self, object, method, handler, false, false, true, hookSecure or false, "Usage: RawHook([object], method, [handler], [hookSecure])")
|
||||||
|
end
|
||||||
|
|
||||||
|
--- SecureHook a function or a method on an object.
|
||||||
|
-- This function is a wrapper around the `hooksecurefunc` function in the WoW API. Using AceHook
|
||||||
|
-- extends the functionality of secure hooks, and adds the ability to unhook once the hook isn't
|
||||||
|
-- required anymore, or the addon is being disabled.\\
|
||||||
|
-- Secure Hooks should be used if the secure-status of the function is vital to its function,
|
||||||
|
-- and taint would block execution. Secure Hooks are always called after the original function was called
|
||||||
|
-- ("Post Hook"), and you cannot modify the arguments, return values or control the execution.
|
||||||
|
-- @paramsig [object], method, [handler]
|
||||||
|
-- @param object The object to hook a method from
|
||||||
|
-- @param method If object was specified, the name of the method, or the name of the function to hook.
|
||||||
|
-- @param handler The handler for the hook, a funcref or a method name. (Defaults to the name of the hooked function)
|
||||||
|
function AceHook:SecureHook(object, method, handler)
|
||||||
|
if type(object) == "string" then
|
||||||
|
method, handler, object = object, method, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
hook(self, object, method, handler, false, true, false, false, "Usage: SecureHook([object], method, [handler])")
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Hook a script handler on a frame.
|
||||||
|
-- The hook created will be a "safe hook", that means that your handler will be called
|
||||||
|
-- before the hooked script ("Pre-Hook"), and you don't have to call the original function yourself,
|
||||||
|
-- however you cannot stop the execution of the function, or modify any of the arguments/return values.\\
|
||||||
|
-- This is the frame script equivalent of the :Hook safe-hook. It would typically be used to be notified
|
||||||
|
-- when a certain event happens to a frame.
|
||||||
|
-- @paramsig frame, script, [handler]
|
||||||
|
-- @param frame The Frame to hook the script on
|
||||||
|
-- @param script The script to hook
|
||||||
|
-- @param handler The handler for the hook, a funcref or a method name. (Defaults to the name of the hooked script)
|
||||||
|
-- @usage
|
||||||
|
-- -- create an addon with AceHook embeded
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("HookDemo", "AceHook-3.0")
|
||||||
|
--
|
||||||
|
-- function MyAddon:OnEnable()
|
||||||
|
-- -- Hook the OnShow of FriendsFrame
|
||||||
|
-- self:HookScript(FriendsFrame, "OnShow", "FriendsFrameOnShow")
|
||||||
|
-- end
|
||||||
|
--
|
||||||
|
-- function MyAddon:FriendsFrameOnShow(frame)
|
||||||
|
-- print("The FriendsFrame was shown!")
|
||||||
|
-- end
|
||||||
|
function AceHook:HookScript(frame, script, handler)
|
||||||
|
hook(self, frame, script, handler, true, false, false, false, "Usage: HookScript(object, method, [handler])")
|
||||||
|
end
|
||||||
|
|
||||||
|
--- RawHook a script handler on a frame.
|
||||||
|
-- The hook created will be a "raw hook", that means that your handler will completly replace
|
||||||
|
-- the original script, and your handler has to call the original script (or not, depending on your intentions).\\
|
||||||
|
-- The original script will be stored in `self.hooks[frame][script]`.\\
|
||||||
|
-- This type of hook can be used for all purposes, and is usually the most common case when you need to modify arguments
|
||||||
|
-- or want to control execution of the original script.
|
||||||
|
-- @paramsig frame, script, [handler]
|
||||||
|
-- @param frame The Frame to hook the script on
|
||||||
|
-- @param script The script to hook
|
||||||
|
-- @param handler The handler for the hook, a funcref or a method name. (Defaults to the name of the hooked script)
|
||||||
|
-- @usage
|
||||||
|
-- -- create an addon with AceHook embeded
|
||||||
|
-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("HookDemo", "AceHook-3.0")
|
||||||
|
--
|
||||||
|
-- function MyAddon:OnEnable()
|
||||||
|
-- -- Hook the OnShow of FriendsFrame
|
||||||
|
-- self:RawHookScript(FriendsFrame, "OnShow", "FriendsFrameOnShow")
|
||||||
|
-- end
|
||||||
|
--
|
||||||
|
-- function MyAddon:FriendsFrameOnShow(frame)
|
||||||
|
-- -- Call the original function
|
||||||
|
-- self.hooks[frame].OnShow(frame)
|
||||||
|
-- -- Do our processing
|
||||||
|
-- -- .. stuff
|
||||||
|
-- end
|
||||||
|
function AceHook:RawHookScript(frame, script, handler)
|
||||||
|
hook(self, frame, script, handler, true, false, true, false, "Usage: RawHookScript(object, method, [handler])")
|
||||||
|
end
|
||||||
|
|
||||||
|
--- SecureHook a script handler on a frame.
|
||||||
|
-- This function is a wrapper around the `frame:HookScript` function in the WoW API. Using AceHook
|
||||||
|
-- extends the functionality of secure hooks, and adds the ability to unhook once the hook isn't
|
||||||
|
-- required anymore, or the addon is being disabled.\\
|
||||||
|
-- Secure Hooks should be used if the secure-status of the function is vital to its function,
|
||||||
|
-- and taint would block execution. Secure Hooks are always called after the original function was called
|
||||||
|
-- ("Post Hook"), and you cannot modify the arguments, return values or control the execution.
|
||||||
|
-- @paramsig frame, script, [handler]
|
||||||
|
-- @param frame The Frame to hook the script on
|
||||||
|
-- @param script The script to hook
|
||||||
|
-- @param handler The handler for the hook, a funcref or a method name. (Defaults to the name of the hooked script)
|
||||||
|
function AceHook:SecureHookScript(frame, script, handler)
|
||||||
|
hook(self, frame, script, handler, true, true, false, false, "Usage: SecureHookScript(object, method, [handler])")
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Unhook from the specified function, method or script.
|
||||||
|
-- @paramsig [obj], method
|
||||||
|
-- @param obj The object or frame to unhook from
|
||||||
|
-- @param method The name of the method, function or script to unhook from.
|
||||||
|
function AceHook:Unhook(obj, method)
|
||||||
|
local usage = "Usage: Unhook([obj], method)"
|
||||||
|
if type(obj) == "string" then
|
||||||
|
method, obj = obj, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
if obj and type(obj) ~= "table" then
|
||||||
|
error(format("%s: 'obj' - expecting nil or table got %s", usage, type(obj)), 2)
|
||||||
|
end
|
||||||
|
if type(method) ~= "string" then
|
||||||
|
error(format("%s: 'method' - expeting string got %s", usage, type(method)), 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
local uid
|
||||||
|
if obj then
|
||||||
|
uid = registry[self][obj] and registry[self][obj][method]
|
||||||
|
else
|
||||||
|
uid = registry[self][method]
|
||||||
|
end
|
||||||
|
|
||||||
|
if not uid or not actives[uid] then
|
||||||
|
-- Declining to error on an unneeded unhook since the end effect is the same and this would just be annoying.
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
actives[uid], handlers[uid] = nil, nil
|
||||||
|
|
||||||
|
if obj then
|
||||||
|
registry[self][obj][method] = nil
|
||||||
|
registry[self][obj] = next(registry[self][obj]) and registry[self][obj] or nil
|
||||||
|
|
||||||
|
-- if the hook reference doesnt exist, then its a secure hook, just bail out and dont do any unhooking
|
||||||
|
if not self.hooks[obj] or not self.hooks[obj][method] then return true end
|
||||||
|
|
||||||
|
if scripts[uid] and obj:GetScript(method) == uid then -- unhooks scripts
|
||||||
|
obj:SetScript(method, self.hooks[obj][method] ~= donothing and self.hooks[obj][method] or nil)
|
||||||
|
scripts[uid] = nil
|
||||||
|
elseif obj and self.hooks[obj] and self.hooks[obj][method] and obj[method] == uid then -- unhooks methods
|
||||||
|
obj[method] = self.hooks[obj][method]
|
||||||
|
end
|
||||||
|
|
||||||
|
self.hooks[obj][method] = nil
|
||||||
|
self.hooks[obj] = next(self.hooks[obj]) and self.hooks[obj] or nil
|
||||||
|
else
|
||||||
|
registry[self][method] = nil
|
||||||
|
|
||||||
|
-- if self.hooks[method] doesn't exist, then this is a SecureHook, just bail out
|
||||||
|
if not self.hooks[method] then return true end
|
||||||
|
|
||||||
|
if self.hooks[method] and _G[method] == uid then -- unhooks functions
|
||||||
|
_G[method] = self.hooks[method]
|
||||||
|
end
|
||||||
|
|
||||||
|
self.hooks[method] = nil
|
||||||
|
end
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Unhook all existing hooks for this addon.
|
||||||
|
function AceHook:UnhookAll()
|
||||||
|
for key, value in pairs(registry[self]) do
|
||||||
|
if type(key) == "table" then
|
||||||
|
for method in pairs(value) do
|
||||||
|
self:Unhook(key, method)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
self:Unhook(key)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Check if the specific function, method or script is already hooked.
|
||||||
|
-- @paramsig [obj], method
|
||||||
|
-- @param obj The object or frame to unhook from
|
||||||
|
-- @param method The name of the method, function or script to unhook from.
|
||||||
|
function AceHook:IsHooked(obj, method)
|
||||||
|
-- we don't check if registry[self] exists, this is done by evil magicks in the metatable
|
||||||
|
if type(obj) == "string" then
|
||||||
|
if registry[self][obj] and actives[registry[self][obj]] then
|
||||||
|
return true, handlers[registry[self][obj]]
|
||||||
|
end
|
||||||
|
else
|
||||||
|
if registry[self][obj] and registry[self][obj][method] and actives[registry[self][obj][method]] then
|
||||||
|
return true, handlers[registry[self][obj][method]]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Upgrade our old embeded
|
||||||
|
for target, v in pairs( AceHook.embeded ) do
|
||||||
|
AceHook:Embed( target )
|
||||||
|
end
|
@ -0,0 +1,4 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
<Script file="AceHook-3.0.lua"/>
|
||||||
|
</Ui>
|
@ -0,0 +1,137 @@
|
|||||||
|
--- **AceLocale-3.0** manages localization in addons, allowing for multiple locale to be registered with fallback to the base locale for untranslated strings.
|
||||||
|
-- @class file
|
||||||
|
-- @name AceLocale-3.0
|
||||||
|
-- @release $Id: AceLocale-3.0.lua 1035 2011-07-09 03:20:13Z kaelten $
|
||||||
|
local MAJOR,MINOR = "AceLocale-3.0", 6
|
||||||
|
|
||||||
|
local AceLocale, oldminor = LibStub:NewLibrary(MAJOR, MINOR)
|
||||||
|
|
||||||
|
if not AceLocale then return end -- no upgrade needed
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local assert, tostring, error = assert, tostring, error
|
||||||
|
local getmetatable, setmetatable, rawset, rawget = getmetatable, setmetatable, rawset, rawget
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: GAME_LOCALE, geterrorhandler
|
||||||
|
|
||||||
|
local gameLocale = GetLocale()
|
||||||
|
if gameLocale == "enGB" then
|
||||||
|
gameLocale = "enUS"
|
||||||
|
end
|
||||||
|
|
||||||
|
AceLocale.apps = AceLocale.apps or {} -- array of ["AppName"]=localetableref
|
||||||
|
AceLocale.appnames = AceLocale.appnames or {} -- array of [localetableref]="AppName"
|
||||||
|
|
||||||
|
-- This metatable is used on all tables returned from GetLocale
|
||||||
|
local readmeta = {
|
||||||
|
__index = function(self, key) -- requesting totally unknown entries: fire off a nonbreaking error and return key
|
||||||
|
rawset(self, key, key) -- only need to see the warning once, really
|
||||||
|
geterrorhandler()(MAJOR..": "..tostring(AceLocale.appnames[self])..": Missing entry for '"..tostring(key).."'")
|
||||||
|
return key
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
-- This metatable is used on all tables returned from GetLocale if the silent flag is true, it does not issue a warning on unknown keys
|
||||||
|
local readmetasilent = {
|
||||||
|
__index = function(self, key) -- requesting totally unknown entries: return key
|
||||||
|
rawset(self, key, key) -- only need to invoke this function once
|
||||||
|
return key
|
||||||
|
end
|
||||||
|
}
|
||||||
|
|
||||||
|
-- Remember the locale table being registered right now (it gets set by :NewLocale())
|
||||||
|
-- NOTE: Do never try to register 2 locale tables at once and mix their definition.
|
||||||
|
local registering
|
||||||
|
|
||||||
|
-- local assert false function
|
||||||
|
local assertfalse = function() assert(false) end
|
||||||
|
|
||||||
|
-- This metatable proxy is used when registering nondefault locales
|
||||||
|
local writeproxy = setmetatable({}, {
|
||||||
|
__newindex = function(self, key, value)
|
||||||
|
rawset(registering, key, value == true and key or value) -- assigning values: replace 'true' with key string
|
||||||
|
end,
|
||||||
|
__index = assertfalse
|
||||||
|
})
|
||||||
|
|
||||||
|
-- This metatable proxy is used when registering the default locale.
|
||||||
|
-- It refuses to overwrite existing values
|
||||||
|
-- Reason 1: Allows loading locales in any order
|
||||||
|
-- Reason 2: If 2 modules have the same string, but only the first one to be
|
||||||
|
-- loaded has a translation for the current locale, the translation
|
||||||
|
-- doesn't get overwritten.
|
||||||
|
--
|
||||||
|
local writedefaultproxy = setmetatable({}, {
|
||||||
|
__newindex = function(self, key, value)
|
||||||
|
if not rawget(registering, key) then
|
||||||
|
rawset(registering, key, value == true and key or value)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
__index = assertfalse
|
||||||
|
})
|
||||||
|
|
||||||
|
--- Register a new locale (or extend an existing one) for the specified application.
|
||||||
|
-- :NewLocale will return a table you can fill your locale into, or nil if the locale isn't needed for the players
|
||||||
|
-- game locale.
|
||||||
|
-- @paramsig application, locale[, isDefault[, silent]]
|
||||||
|
-- @param application Unique name of addon / module
|
||||||
|
-- @param locale Name of the locale to register, e.g. "enUS", "deDE", etc.
|
||||||
|
-- @param isDefault If this is the default locale being registered (your addon is written in this language, generally enUS)
|
||||||
|
-- @param silent If true, the locale will not issue warnings for missing keys. Must be set on the first locale registered. If set to "raw", nils will be returned for unknown keys (no metatable used).
|
||||||
|
-- @usage
|
||||||
|
-- -- enUS.lua
|
||||||
|
-- local L = LibStub("AceLocale-3.0"):NewLocale("TestLocale", "enUS", true)
|
||||||
|
-- L["string1"] = true
|
||||||
|
--
|
||||||
|
-- -- deDE.lua
|
||||||
|
-- local L = LibStub("AceLocale-3.0"):NewLocale("TestLocale", "deDE")
|
||||||
|
-- if not L then return end
|
||||||
|
-- L["string1"] = "Zeichenkette1"
|
||||||
|
-- @return Locale Table to add localizations to, or nil if the current locale is not required.
|
||||||
|
function AceLocale:NewLocale(application, locale, isDefault, silent)
|
||||||
|
|
||||||
|
-- GAME_LOCALE allows translators to test translations of addons without having that wow client installed
|
||||||
|
local gameLocale = GAME_LOCALE or gameLocale
|
||||||
|
|
||||||
|
local app = AceLocale.apps[application]
|
||||||
|
|
||||||
|
if silent and app and getmetatable(app) ~= readmetasilent then
|
||||||
|
geterrorhandler()("Usage: NewLocale(application, locale[, isDefault[, silent]]): 'silent' must be specified for the first locale registered")
|
||||||
|
end
|
||||||
|
|
||||||
|
if not app then
|
||||||
|
if silent=="raw" then
|
||||||
|
app = {}
|
||||||
|
else
|
||||||
|
app = setmetatable({}, silent and readmetasilent or readmeta)
|
||||||
|
end
|
||||||
|
AceLocale.apps[application] = app
|
||||||
|
AceLocale.appnames[app] = application
|
||||||
|
end
|
||||||
|
|
||||||
|
if locale ~= gameLocale and not isDefault then
|
||||||
|
return -- nop, we don't need these translations
|
||||||
|
end
|
||||||
|
|
||||||
|
registering = app -- remember globally for writeproxy and writedefaultproxy
|
||||||
|
|
||||||
|
if isDefault then
|
||||||
|
return writedefaultproxy
|
||||||
|
end
|
||||||
|
|
||||||
|
return writeproxy
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Returns localizations for the current locale (or default locale if translations are missing).
|
||||||
|
-- Errors if nothing is registered (spank developer, not just a missing translation)
|
||||||
|
-- @param application Unique name of addon / module
|
||||||
|
-- @param silent If true, the locale is optional, silently return nil if it's not found (defaults to false, optional)
|
||||||
|
-- @return The locale table for the current language.
|
||||||
|
function AceLocale:GetLocale(application, silent)
|
||||||
|
if not silent and not AceLocale.apps[application] then
|
||||||
|
error("Usage: GetLocale(application[, silent]): 'application' - No locales registered for '"..tostring(application).."'", 2)
|
||||||
|
end
|
||||||
|
return AceLocale.apps[application]
|
||||||
|
end
|
@ -0,0 +1,4 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
<Script file="AceLocale-3.0.lua"/>
|
||||||
|
</Ui>
|
@ -0,0 +1,212 @@
|
|||||||
|
--[[ $Id: CallbackHandler-1.0.lua 22 2018-07-21 14:17:22Z nevcairiel $ ]]
|
||||||
|
local MAJOR, MINOR = "CallbackHandler-1.0", 7
|
||||||
|
local CallbackHandler = LibStub:NewLibrary(MAJOR, MINOR)
|
||||||
|
|
||||||
|
if not CallbackHandler then return end -- No upgrade needed
|
||||||
|
|
||||||
|
local meta = {__index = function(tbl, key) tbl[key] = {} return tbl[key] end}
|
||||||
|
|
||||||
|
-- Lua APIs
|
||||||
|
local tconcat = table.concat
|
||||||
|
local assert, error, loadstring = assert, error, loadstring
|
||||||
|
local setmetatable, rawset, rawget = setmetatable, rawset, rawget
|
||||||
|
local next, select, pairs, type, tostring = next, select, pairs, type, tostring
|
||||||
|
|
||||||
|
-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
|
||||||
|
-- List them here for Mikk's FindGlobals script
|
||||||
|
-- GLOBALS: geterrorhandler
|
||||||
|
|
||||||
|
local xpcall = xpcall
|
||||||
|
|
||||||
|
local function errorhandler(err)
|
||||||
|
return geterrorhandler()(err)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function Dispatch(handlers, ...)
|
||||||
|
local index, method = next(handlers)
|
||||||
|
if not method then return end
|
||||||
|
repeat
|
||||||
|
xpcall(method, errorhandler, ...)
|
||||||
|
index, method = next(handlers, index)
|
||||||
|
until not method
|
||||||
|
end
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------
|
||||||
|
-- CallbackHandler:New
|
||||||
|
--
|
||||||
|
-- target - target object to embed public APIs in
|
||||||
|
-- RegisterName - name of the callback registration API, default "RegisterCallback"
|
||||||
|
-- UnregisterName - name of the callback unregistration API, default "UnregisterCallback"
|
||||||
|
-- UnregisterAllName - name of the API to unregister all callbacks, default "UnregisterAllCallbacks". false == don't publish this API.
|
||||||
|
|
||||||
|
function CallbackHandler:New(target, RegisterName, UnregisterName, UnregisterAllName)
|
||||||
|
|
||||||
|
RegisterName = RegisterName or "RegisterCallback"
|
||||||
|
UnregisterName = UnregisterName or "UnregisterCallback"
|
||||||
|
if UnregisterAllName==nil then -- false is used to indicate "don't want this method"
|
||||||
|
UnregisterAllName = "UnregisterAllCallbacks"
|
||||||
|
end
|
||||||
|
|
||||||
|
-- we declare all objects and exported APIs inside this closure to quickly gain access
|
||||||
|
-- to e.g. function names, the "target" parameter, etc
|
||||||
|
|
||||||
|
|
||||||
|
-- Create the registry object
|
||||||
|
local events = setmetatable({}, meta)
|
||||||
|
local registry = { recurse=0, events=events }
|
||||||
|
|
||||||
|
-- registry:Fire() - fires the given event/message into the registry
|
||||||
|
function registry:Fire(eventname, ...)
|
||||||
|
if not rawget(events, eventname) or not next(events[eventname]) then return end
|
||||||
|
local oldrecurse = registry.recurse
|
||||||
|
registry.recurse = oldrecurse + 1
|
||||||
|
|
||||||
|
Dispatch(events[eventname], eventname, ...)
|
||||||
|
|
||||||
|
registry.recurse = oldrecurse
|
||||||
|
|
||||||
|
if registry.insertQueue and oldrecurse==0 then
|
||||||
|
-- Something in one of our callbacks wanted to register more callbacks; they got queued
|
||||||
|
for eventname,callbacks in pairs(registry.insertQueue) do
|
||||||
|
local first = not rawget(events, eventname) or not next(events[eventname]) -- test for empty before. not test for one member after. that one member may have been overwritten.
|
||||||
|
for self,func in pairs(callbacks) do
|
||||||
|
events[eventname][self] = func
|
||||||
|
-- fire OnUsed callback?
|
||||||
|
if first and registry.OnUsed then
|
||||||
|
registry.OnUsed(registry, target, eventname)
|
||||||
|
first = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
registry.insertQueue = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Registration of a callback, handles:
|
||||||
|
-- self["method"], leads to self["method"](self, ...)
|
||||||
|
-- self with function ref, leads to functionref(...)
|
||||||
|
-- "addonId" (instead of self) with function ref, leads to functionref(...)
|
||||||
|
-- all with an optional arg, which, if present, gets passed as first argument (after self if present)
|
||||||
|
target[RegisterName] = function(self, eventname, method, ... --[[actually just a single arg]])
|
||||||
|
if type(eventname) ~= "string" then
|
||||||
|
error("Usage: "..RegisterName.."(eventname, method[, arg]): 'eventname' - string expected.", 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
method = method or eventname
|
||||||
|
|
||||||
|
local first = not rawget(events, eventname) or not next(events[eventname]) -- test for empty before. not test for one member after. that one member may have been overwritten.
|
||||||
|
|
||||||
|
if type(method) ~= "string" and type(method) ~= "function" then
|
||||||
|
error("Usage: "..RegisterName.."(\"eventname\", \"methodname\"): 'methodname' - string or function expected.", 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
local regfunc
|
||||||
|
|
||||||
|
if type(method) == "string" then
|
||||||
|
-- self["method"] calling style
|
||||||
|
if type(self) ~= "table" then
|
||||||
|
error("Usage: "..RegisterName.."(\"eventname\", \"methodname\"): self was not a table?", 2)
|
||||||
|
elseif self==target then
|
||||||
|
error("Usage: "..RegisterName.."(\"eventname\", \"methodname\"): do not use Library:"..RegisterName.."(), use your own 'self'", 2)
|
||||||
|
elseif type(self[method]) ~= "function" then
|
||||||
|
error("Usage: "..RegisterName.."(\"eventname\", \"methodname\"): 'methodname' - method '"..tostring(method).."' not found on self.", 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
if select("#",...)>=1 then -- this is not the same as testing for arg==nil!
|
||||||
|
local arg=select(1,...)
|
||||||
|
regfunc = function(...) self[method](self,arg,...) end
|
||||||
|
else
|
||||||
|
regfunc = function(...) self[method](self,...) end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
-- function ref with self=object or self="addonId" or self=thread
|
||||||
|
if type(self)~="table" and type(self)~="string" and type(self)~="thread" then
|
||||||
|
error("Usage: "..RegisterName.."(self or \"addonId\", eventname, method): 'self or addonId': table or string or thread expected.", 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
if select("#",...)>=1 then -- this is not the same as testing for arg==nil!
|
||||||
|
local arg=select(1,...)
|
||||||
|
regfunc = function(...) method(arg,...) end
|
||||||
|
else
|
||||||
|
regfunc = method
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
if events[eventname][self] or registry.recurse<1 then
|
||||||
|
-- if registry.recurse<1 then
|
||||||
|
-- we're overwriting an existing entry, or not currently recursing. just set it.
|
||||||
|
events[eventname][self] = regfunc
|
||||||
|
-- fire OnUsed callback?
|
||||||
|
if registry.OnUsed and first then
|
||||||
|
registry.OnUsed(registry, target, eventname)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
-- we're currently processing a callback in this registry, so delay the registration of this new entry!
|
||||||
|
-- yes, we're a bit wasteful on garbage, but this is a fringe case, so we're picking low implementation overhead over garbage efficiency
|
||||||
|
registry.insertQueue = registry.insertQueue or setmetatable({},meta)
|
||||||
|
registry.insertQueue[eventname][self] = regfunc
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Unregister a callback
|
||||||
|
target[UnregisterName] = function(self, eventname)
|
||||||
|
if not self or self==target then
|
||||||
|
error("Usage: "..UnregisterName.."(eventname): bad 'self'", 2)
|
||||||
|
end
|
||||||
|
if type(eventname) ~= "string" then
|
||||||
|
error("Usage: "..UnregisterName.."(eventname): 'eventname' - string expected.", 2)
|
||||||
|
end
|
||||||
|
if rawget(events, eventname) and events[eventname][self] then
|
||||||
|
events[eventname][self] = nil
|
||||||
|
-- Fire OnUnused callback?
|
||||||
|
if registry.OnUnused and not next(events[eventname]) then
|
||||||
|
registry.OnUnused(registry, target, eventname)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if registry.insertQueue and rawget(registry.insertQueue, eventname) and registry.insertQueue[eventname][self] then
|
||||||
|
registry.insertQueue[eventname][self] = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- OPTIONAL: Unregister all callbacks for given selfs/addonIds
|
||||||
|
if UnregisterAllName then
|
||||||
|
target[UnregisterAllName] = function(...)
|
||||||
|
if select("#",...)<1 then
|
||||||
|
error("Usage: "..UnregisterAllName.."([whatFor]): missing 'self' or \"addonId\" to unregister events for.", 2)
|
||||||
|
end
|
||||||
|
if select("#",...)==1 and ...==target then
|
||||||
|
error("Usage: "..UnregisterAllName.."([whatFor]): supply a meaningful 'self' or \"addonId\"", 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
for i=1,select("#",...) do
|
||||||
|
local self = select(i,...)
|
||||||
|
if registry.insertQueue then
|
||||||
|
for eventname, callbacks in pairs(registry.insertQueue) do
|
||||||
|
if callbacks[self] then
|
||||||
|
callbacks[self] = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
for eventname, callbacks in pairs(events) do
|
||||||
|
if callbacks[self] then
|
||||||
|
callbacks[self] = nil
|
||||||
|
-- Fire OnUnused callback?
|
||||||
|
if registry.OnUnused and not next(callbacks) then
|
||||||
|
registry.OnUnused(registry, target, eventname)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
return registry
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
-- CallbackHandler purposefully does NOT do explicit embedding. Nor does it
|
||||||
|
-- try to upgrade old implicit embeds since the system is selfcontained and
|
||||||
|
-- relies on closures to work.
|
||||||
|
|
@ -0,0 +1,4 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
<Script file="CallbackHandler-1.0.lua"/>
|
||||||
|
</Ui>
|
@ -0,0 +1,30 @@
|
|||||||
|
-- LibStub is a simple versioning stub meant for use in Libraries. http://www.wowace.com/wiki/LibStub for more info
|
||||||
|
-- LibStub is hereby placed in the Public Domain Credits: Kaelten, Cladhaire, ckknight, Mikk, Ammo, Nevcairiel, joshborke
|
||||||
|
local LIBSTUB_MAJOR, LIBSTUB_MINOR = "LibStub", 2 -- NEVER MAKE THIS AN SVN REVISION! IT NEEDS TO BE USABLE IN ALL REPOS!
|
||||||
|
local LibStub = _G[LIBSTUB_MAJOR]
|
||||||
|
|
||||||
|
if not LibStub or LibStub.minor < LIBSTUB_MINOR then
|
||||||
|
LibStub = LibStub or {libs = {}, minors = {} }
|
||||||
|
_G[LIBSTUB_MAJOR] = LibStub
|
||||||
|
LibStub.minor = LIBSTUB_MINOR
|
||||||
|
|
||||||
|
function LibStub:NewLibrary(major, minor)
|
||||||
|
assert(type(major) == "string", "Bad argument #2 to `NewLibrary' (string expected)")
|
||||||
|
minor = assert(tonumber(strmatch(minor, "%d+")), "Minor version must either be a number or contain a number.")
|
||||||
|
|
||||||
|
local oldminor = self.minors[major]
|
||||||
|
if oldminor and oldminor >= minor then return nil end
|
||||||
|
self.minors[major], self.libs[major] = minor, self.libs[major] or {}
|
||||||
|
return self.libs[major], oldminor
|
||||||
|
end
|
||||||
|
|
||||||
|
function LibStub:GetLibrary(major, silent)
|
||||||
|
if not self.libs[major] and not silent then
|
||||||
|
error(("Cannot find a library instance of %q."):format(tostring(major)), 2)
|
||||||
|
end
|
||||||
|
return self.libs[major], self.minors[major]
|
||||||
|
end
|
||||||
|
|
||||||
|
function LibStub:IterateLibraries() return pairs(self.libs) end
|
||||||
|
setmetatable(LibStub, { __call = LibStub.GetLibrary })
|
||||||
|
end
|
@ -0,0 +1,9 @@
|
|||||||
|
## Interface: 20400
|
||||||
|
## Title: Lib: LibStub
|
||||||
|
## Notes: Universal Library Stub
|
||||||
|
## Credits: Kaelten, Cladhaire, ckknight, Mikk, Ammo, Nevcairiel
|
||||||
|
## X-Website: http://jira.wowace.com/browse/LS
|
||||||
|
## X-Category: Library
|
||||||
|
## X-License: Public Domain
|
||||||
|
|
||||||
|
LibStub.lua
|
@ -0,0 +1,5 @@
|
|||||||
|
<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
|
||||||
|
..\FrameXML\UI.xsd">
|
||||||
|
<Script file="localization\localisation.lua"/>
|
||||||
|
<Script file="localization\localisation-de.lua"/>
|
||||||
|
</Ui>
|
@ -0,0 +1,22 @@
|
|||||||
|
--[[
|
||||||
|
Localization.lua
|
||||||
|
Translations for Dominos
|
||||||
|
|
||||||
|
German language
|
||||||
|
--]]
|
||||||
|
|
||||||
|
local L = LibStub('AceLocale-3.0'):NewLocale('Grichelde', 'deDE')
|
||||||
|
if not L then return end
|
||||||
|
|
||||||
|
--system messages
|
||||||
|
L.NewPlayer = 'Neues Profil f\195\188r %s erstellt'
|
||||||
|
L.Updated = 'Aktualisiert auf v%s'
|
||||||
|
|
||||||
|
--profiles
|
||||||
|
L.ProfileCreated = 'Neues Profil "%s" erstellt'
|
||||||
|
L.ProfileLoaded = 'Profil auf "%s" festgelegt'
|
||||||
|
L.ProfileDeleted = 'Profil "%s" gel\195\182scht'
|
||||||
|
L.ProfileCopied = 'Einstellungen von "%s" kopiert'
|
||||||
|
L.ProfileReset = 'Profil "%s" zur\195\188ckgesetzt'
|
||||||
|
L.CantDeleteCurrentProfile = 'Das aktuelle Profil kann nicht gel\195\182scht werden'
|
||||||
|
L.InvalidProfile = 'Ung\195\188ltiges Profil "%s"'
|
Loading…
Reference in New Issue