vanilla-wow-addons – Rev 1

Subversion Repositories:
Rev:
--[[
Name: AceAddon-2.0
Revision: $Rev: 14832 $
Developed by: The Ace Development Team (http://www.wowace.com/index.php/The_Ace_Development_Team)
Inspired By: Ace 1.x by Turan (turan@gryphon.com)
Website: http://www.wowace.com/
Documentation: http://www.wowace.com/index.php/AceAddon-2.0
SVN: http://svn.wowace.com/root/trunk/Ace2/AceAddon-2.0
Description: Base for all Ace addons to inherit from.
Dependencies: AceLibrary, AceOO-2.0, AceEvent-2.0, (optional) AceConsole-2.0
]]

local MAJOR_VERSION = "AceAddon-2.0"
local MINOR_VERSION = "$Revision: 14832 $"

-- This ensures the code is only executed if the libary doesn't already exist, or is a newer version
if not AceLibrary then error(MAJOR_VERSION .. " requires AceLibrary.") end
if not AceLibrary:IsNewVersion(MAJOR_VERSION, MINOR_VERSION) then return end

if not AceLibrary:HasInstance("AceOO-2.0") then error(MAJOR_VERSION .. " requires AceOO-2.0.") end

-- Localization
local STANDBY, TITLE, NOTES, VERSION, AUTHOR, DATE, CATEGORY, EMAIL, WEBSITE, CATEGORIES, ABOUT, PRINT_ADDON_INFO
if GetLocale() == "deDE" then
        STANDBY = "|cffff5050(Standby)|r" -- capitalized

        TITLE = "Titel"
        NOTES = "Anmerkung"
        VERSION = "Version"
        AUTHOR = "Autor"
        DATE = "Datum"
        CATEGORY = "Kategorie"
        EMAIL = "E-mail"
        WEBSITE = "Webseite"
                 
        ABOUT = "\195\188ber"
        PRINT_ADDON_INFO = "Gibt Addondaten aus"

        CATEGORIES = {
                ["Action Bars"] = "Aktionsleisten",
                ["Auction"] = "Auktion",
                ["Audio"] = "Audio",
                ["Battlegrounds/PvP"] = "Schlachtfeld/PvP",
                ["Buffs"] = "Buffs",
                ["Chat/Communication"] = "Chat/Kommunikation",
                ["Druid"] = "Druide",
                ["Hunter"] = "Jäger",
                ["Mage"] = "Magier",
                ["Paladin"] = "Paladin",
                ["Priest"] = "Priester",
                ["Rogue"] = "Schurke",
                ["Shaman"] = "Schamane",
                ["Warlock"] = "Hexenmeister",
                ["Warrior"] = "Krieger",
                ["Healer"] = "Heiler",
                ["Tank"] = "Tank", -- noone use "Brecher"...
                ["Caster"] = "Caster",
                ["Combat"] = "Kampf",
                ["Compilations"] = "Compilations", -- whats that o_O
                ["Data Export"] = "Datenexport",
                ["Development Tools"] = "Entwicklungs Tools",
                ["Guild"] = "Gilde",
                ["Frame Modification"] = "Frame Modifikation",
                ["Interface Enhancements"] = "Interface Verbesserungen",
                ["Inventory"] = "Inventar",
                ["Library"] = "Library",
                ["Map"] = "Map",
                ["Mail"] = "Mail",
                ["Miscellaneous"] = "Diverses",
                ["Quest"] = "Quest",
                ["Raid"] = "Schlachtzug",
                ["Tradeskill"] = "Handelsf\195\164higkeit",
                ["UnitFrame"] = "UnitFrame",
        }
elseif GetLocale() == "frFR" then
        STANDBY = "|cffff5050(attente)|r"
        
        TITLE = "Titre"
        NOTES = "Notes"
        VERSION = "Version"
        AUTHOR = "Auteur"
        DATE = "Date"
        CATEGORY = "Cat\195\169gorie"
        EMAIL = "E-mail"
        WEBSITE = "Site web"
        
        ABOUT = "A propos"
        PRINT_ADDON_INFO = "Afficher les informations sur l'addon"
        
        CATEGORIES = {
                ["Action Bars"] = "Barres d'action",
                ["Auction"] = "H\195\180tel des ventes",
                ["Audio"] = "Audio",
                ["Battlegrounds/PvP"] = "Champs de bataille/JcJ",
                ["Buffs"] = "Buffs",
                ["Chat/Communication"] = "Chat/Communication",
                ["Druid"] = "Druide",
                ["Hunter"] = "Chasseur",
                ["Mage"] = "Mage",
                ["Paladin"] = "Paladin",
                ["Priest"] = "Pr\195\170tre",
                ["Rogue"] = "Voleur",
                ["Shaman"] = "Chaman",
                ["Warlock"] = "D\195\169moniste",
                ["Warrior"] = "Guerrier",
                ["Healer"] = "Soigneur",
                ["Tank"] = "Tank",
                ["Caster"] = "Casteur",
                ["Combat"] = "Combat",
                ["Compilations"] = "Compilations",
                ["Data Export"] = "Exportation de donn\195\169es",
                ["Development Tools"] = "Outils de d\195\169veloppement",
                ["Guild"] = "Guilde",
                ["Frame Modification"] = "Modification des fen\195\170tres",
                ["Interface Enhancements"] = "Am\195\169liorations de l'interface",
                ["Inventory"] = "Inventaire",
                ["Library"] = "Biblioth\195\168ques",
                ["Map"] = "Carte",
                ["Mail"] = "Courrier",
                ["Miscellaneous"] = "Divers",
                ["Quest"] = "Qu\195\170tes",
                ["Raid"] = "Raid",
                ["Tradeskill"] = "M\195\169tiers",
                ["UnitFrame"] = "Fen\195\170tres d'unit\195\169",
        }
elseif GetLocale() == "koKR" then
        STANDBY = "|cffff5050(사용가능)|r"
        
        TITLE = "제목"
        NOTES = "노트"
        VERSION = "버전"
        AUTHOR = "저작자"
        DATE = "날짜"
        CATEGORY = "분류"
        EMAIL = "E-mail"
        WEBSITE = "웹사이트"
        
        ABOUT = "정보"
        PRINT_ADDON_INFO = "애드온 정보 출력"
        
        CATEGORIES = {
                ["Action Bars"] = "액션바",
                ["Auction"] = "경매",
                ["Audio"] = "음향",
                ["Battlegrounds/PvP"] = "전장/PvP",
                ["Buffs"] = "버프",
                ["Chat/Communication"] = "대화/의사소통",
                ["Druid"] = "드루이드",
                ["Hunter"] = "사냥꾼",
                ["Mage"] = "마법사",
                ["Paladin"] = "성기사",
                ["Priest"] = "사제",
                ["Rogue"] = "도적",
                ["Shaman"] = "주술사",
                ["Warlock"] = "흑마법사",
                ["Warrior"] = "전사",
                ["Healer"] = "힐러",
                ["Tank"] = "탱커",
                ["Caster"] = "캐스터",
                ["Combat"] = "전투",
                ["Compilations"] = "복합",
                ["Data Export"] = "자료 출력",
                ["Development Tools"] = "개발 도구",
                ["Guild"] = "길드",
                ["Frame Modification"] = "구조 변경",
                ["Interface Enhancements"] = "인터페이스 강화",
                ["Inventory"] = "인벤토리",
                ["Library"] = "라이브러리",
                ["Map"] = "지도",
                ["Mail"] = "우편",
                ["Miscellaneous"] = "기타",
                ["Quest"] = "퀘스트",
                ["Raid"] = "공격대",
                ["Tradeskill"] = "전문기술",
                ["UnitFrame"] = "유닛 프레임",
        }
elseif GetLocale() == "zhTW" then
        STANDBY = "|cffff5050(待命)|r"
        
        TITLE = "標題"
        NOTES = "註記"
        VERSION = "版本"
        AUTHOR = "作者"
        DATE = "日期"
        CATEGORY = "類別"
        EMAIL = "E-mail"
        WEBSITE = "網站"
        
        ABOUT = "關於"
        PRINT_ADDON_INFO = "顯示插件資訊"
        
        CATEGORIES = {
                ["Action Bars"] = "動作列",
                ["Auction"] = "拍賣",
                ["Audio"] = "音樂",
                ["Battlegrounds/PvP"] = "戰場/PvP",
                ["Buffs"] = "增益",
                ["Chat/Communication"] = "聊天/通訊",
                ["Druid"] = "德魯伊",
                ["Hunter"] = "獵人",
                ["Mage"] = "法師",
                ["Paladin"] = "聖騎士",
                ["Priest"] = "牧師",
                ["Rogue"] = "盜賊",
                ["Shaman"] = "薩滿",
                ["Warlock"] = "術士",
                ["Warrior"] = "戰士",
                ["Healer"] = "治療者",
                ["Tank"] = "坦克",
                ["Caster"] = "施法者",
                ["Combat"] = "戰鬥",
                ["Compilations"] = "編輯",
                ["Data Export"] = "資料匯出",
                ["Development Tools"] = "開發工具",
                ["Guild"] = "公會",
                ["Frame Modification"] = "框架修改",
                ["Interface Enhancements"] = "介面增強",
                ["Inventory"] = "背包",
                ["Library"] = "資料庫",
                ["Map"] = "地圖",
                ["Mail"] = "郵件",
                ["Miscellaneous"] = "綜合",
                ["Quest"] = "任務",
                ["Raid"] = "團隊",
                ["Tradeskill"] = "商業技能",
                ["UnitFrame"] = "單位框架",
        }
elseif GetLocale() == "zhCN" then
        STANDBY = "|cffff5050(\230\154\130\230\140\130)|r"
        
        TITLE = "\230\160\135\233\162\152"
        NOTES = "\233\153\132\230\179\168"
        VERSION = "\231\137\136\230\156\172"
        AUTHOR = "\228\189\156\232\128\133"
        DATE = "\230\151\165\230\156\159"
        CATEGORY = "\229\136\134\231\177\187"
        EMAIL = "\231\148\181\229\173\144\233\130\174\228\187\182"
        WEBSITE = "\231\189\145\231\171\153"
        
        ABOUT = "\229\133\179\228\186\142"
        PRINT_ADDON_INFO = "\229\141\176\229\136\151\229\135\186\230\143\146\228\187\182\228\191\161\230\129\175"
        
        CATEGORIES = {
                ["Action Bars"] = "\229\138\168\228\189\156\230\157\161",
                ["Auction"] = "\230\139\141\229\141\150",
                ["Audio"] = "\233\159\179\233\162\145",
                ["Battlegrounds/PvP"] = "\230\136\152\229\156\186/PvP",
                ["Buffs"] = "\229\162\158\231\155\138\233\173\148\230\179\149",
                ["Chat/Communication"] = "\232\129\138\229\164\169/\228\186\164\230\181\129",
                ["Druid"] = "\229\190\183\233\178\129\228\188\138",
                ["Hunter"] = "\231\140\142\228\186\186",
                ["Mage"] = "\230\179\149\229\184\136",
                ["Paladin"] = "\229\156\163\233\170\145\229\163\171",
                ["Priest"] = "\231\137\167\229\184\136",
                ["Rogue"] = "\231\155\151\232\180\188",
                ["Shaman"] = "\232\144\168\230\187\161\231\165\173\229\143\184",
                ["Warlock"] = "\230\156\175\229\163\171",
                ["Warrior"] = "\230\136\152\229\163\171",
--              ["Healer"] = "\230\178\187\231\150\151\228\191\157\233\154\156",
--              ["Tank"] = "\232\191\145\230\136\152\230\142\167\229\136\182",
--              ["Caster"] = "\232\191\156\231\168\139\232\190\147\229\135\186",
                ["Combat"] = "\230\136\152\230\150\151",
                ["Compilations"] = "\231\188\150\232\175\145",
                ["Data Export"] = "\230\149\176\230\141\174\229\175\188\229\135\186",
                ["Development Tools"] = "\229\188\128\229\143\145\229\183\165\229\133\183",
                ["Guild"] = "\229\133\172\228\188\154",
                ["Frame Modification"] = "\230\161\134\230\158\182\228\191\174\230\148\185",
                ["Interface Enhancements"] = "\231\149\140\233\157\162\229\162\158\229\188\186",
                ["Inventory"] = "\232\131\140\229\140\133",
                ["Library"] = "\229\186\147",
                ["Map"] = "\229\156\176\229\155\190",
                ["Mail"] = "\233\130\174\228\187\182",
                ["Miscellaneous"] = "\230\157\130\233\161\185",
                ["Quest"] = "\228\187\187\229\138\161",
                ["Raid"] = "\229\155\162\233\152\159",
                ["Tradeskill"] = "\229\149\134\228\184\154\230\138\128\232\131\189",
                ["UnitFrame"] = "\229\164\180\229\131\143\230\161\134\230\158\182",
        }
else -- enUS
        STANDBY = "|cffff5050(standby)|r"
        
        TITLE = "Title"
        NOTES = "Notes"
        VERSION = "Version"
        AUTHOR = "Author"
        DATE = "Date"
        CATEGORY = "Category"
        EMAIL = "E-mail"
        WEBSITE = "Website"
        
        ABOUT = "About"
        PRINT_ADDON_INFO = "Print out addon info"
        
        CATEGORIES = {
                ["Action Bars"] = "Action Bars",
                ["Auction"] = "Auction",
                ["Audio"] = "Audio",
                ["Battlegrounds/PvP"] = "Battlegrounds/PvP",
                ["Buffs"] = "Buffs",
                ["Chat/Communication"] = "Chat/Communication",
                ["Druid"] = "Druid",
                ["Hunter"] = "Hunter",
                ["Mage"] = "Mage",
                ["Paladin"] = "Paladin",
                ["Priest"] = "Priest",
                ["Rogue"] = "Rogue",
                ["Shaman"] = "Shaman",
                ["Warlock"] = "Warlock",
                ["Warrior"] = "Warrior",
                ["Healer"] = "Healer",
                ["Tank"] = "Tank",
                ["Caster"] = "Caster",
                ["Combat"] = "Combat",
                ["Compilations"] = "Compilations",
                ["Data Export"] = "Data Export",
                ["Development Tools"] = "Development Tools",
                ["Guild"] = "Guild",
                ["Frame Modification"] = "Frame Modification",
                ["Interface Enhancements"] = "Interface Enhancements",
                ["Inventory"] = "Inventory",
                ["Library"] = "Library",
                ["Map"] = "Map",
                ["Mail"] = "Mail",
                ["Miscellaneous"] = "Miscellaneous",
                ["Quest"] = "Quest",
                ["Raid"] = "Raid",
                ["Tradeskill"] = "Tradeskill",
                ["UnitFrame"] = "UnitFrame",
        }
end

setmetatable(CATEGORIES, { __index = function(self, key) -- case-insensitive
        local lowerKey = string.lower(key)
        for k,v in CATEGORIES do
                if string.lower(k) == lowerKey then
                        return v
                end
        end
end })

-- Create the library object

local AceOO = AceLibrary("AceOO-2.0")
local AceAddon = AceOO.Class()
local AceEvent
local AceConsole
local AceModuleCore

function AceAddon:ToString()
        return "AceAddon"
end

local function print(text)
        DEFAULT_CHAT_FRAME:AddMessage(text)
end

function AceAddon:ADDON_LOADED(name)
        while table.getn(self.nextAddon) > 0 do
                local addon = table.remove(self.nextAddon, 1)
                table.insert(self.addons, addon)
                if not self.addons[name] then
                        self.addons[name] = addon
                end
                self:InitializeAddon(addon, name)
        end
end

local function RegisterOnEnable(self)
        if DEFAULT_CHAT_FRAME and DEFAULT_CHAT_FRAME.defaultLanguage then -- HACK
                AceAddon.playerLoginFired = true
        end
        if AceAddon.playerLoginFired then
                AceAddon.addonsStarted[self] = true
                if (type(self.IsActive) ~= "function" or self:IsActive()) and (not AceModuleCore or not AceModuleCore:IsModule(self) or AceModuleCore:IsModuleActive(self)) then
                        local current = self.class
                        while true do
                                if current == AceOO.Class then
                                        break
                                end
                                if current.mixins then
                                        for mixin in pairs(current.mixins) do
                                                if type(mixin.OnEmbedEnable) == "function" then
                                                        mixin:OnEmbedEnable(self)
                                                end
                                        end
                                end
                                current = current.super
                        end
                        if type(self.OnEnable) == "function" then
                                self:OnEnable()
                        end
                end
        else
                if not AceAddon.addonsToOnEnable then
                        AceAddon.addonsToOnEnable = {}
                end
                table.insert(AceAddon.addonsToOnEnable, self)
        end
end

local function stripSpaces(text)
        if type(text) == "string" then
                return (string.gsub(string.gsub(text, "^%s*(.-)%s*$", "%1"), "%s%s+", " "))
        end
        return text
end

function AceAddon:InitializeAddon(addon, name)
        if addon.name == nil then
                addon.name = name
        end
        if GetAddOnMetadata then
                -- TOC checks
                if addon.title == nil then
                        addon.title = GetAddOnMetadata(name, "Title")
                        if addon.title then
                                local num = string.find(addon.title, " |cff7fff7f %-Ace2%-|r$")
                                if num then
                                        addon.title = string.sub(addon.title, 1, num - 1)
                                end
                                addon.title = stripSpaces(addon.title)
                        end
                end
                if addon.notes == nil then
                        addon.notes = GetAddOnMetadata(name, "Notes")
                        addon.notes = stripSpaces(addon.notes)
                end
                if addon.version == nil then
                        addon.version = GetAddOnMetadata(name, "Version")
                        if addon.version then
                                if string.find(addon.version, "%$Revision: (%d+) %$") then
                                        addon.version = string.gsub(addon.version, "%$Revision: (%d+) %$", "%1")
                                elseif string.find(addon.version, "%$Rev: (%d+) %$") then
                                        addon.version = string.gsub(addon.version, "%$Rev: (%d+) %$", "%1")
                                elseif string.find(addon.version, "%$LastChangedRevision: (%d+) %$") then
                                        addon.version = string.gsub(addon.version, "%$LastChangedRevision: (%d+) %$", "%1")
                                end
                        end
                        addon.version = stripSpaces(addon.version)
                end
                if addon.author == nil then
                        addon.author = GetAddOnMetadata(name, "Author")
                        addon.author = stripSpaces(addon.author)
                end
                if addon.date == nil then
                        addon.date = GetAddOnMetadata(name, "X-Date") or GetAddOnMetadata(name, "X-ReleaseDate")
                        if addon.date then
                                if string.find(addon.date, "%$Date: (.-) %$") then
                                        addon.date = string.gsub(addon.date, "%$Date: (.-) %$", "%1")
                                elseif string.find(addon.date, "%$LastChangedDate: (.-) %$") then
                                        addon.date = string.gsub(addon.date, "%$LastChangedDate: (.-) %$", "%1")
                                end
                        end
                        addon.date = stripSpaces(addon.date)
                end
                if addon.category == nil then
                        addon.category = GetAddOnMetadata(name, "X-Category")
                        addon.category = stripSpaces(addon.category)
                end
                if addon.email == nil then
                        addon.email = GetAddOnMetadata(name, "X-eMail") or GetAddOnMetadata(name, "X-Email")
                        addon.email = stripSpaces(addon.email)
                end
                if addon.website == nil then
                        addon.website = GetAddOnMetadata(name, "X-Website")
                        addon.website = stripSpaces(addon.website)
                end
        end
        local current = addon.class
        while true do
                if current == AceOO.Class then
                        break
                end
                if current.mixins then
                        for mixin in pairs(current.mixins) do
                                if type(mixin.OnEmbedInitialize) == "function" then
                                        mixin:OnEmbedInitialize(addon, name)
                                end
                        end
                end
                current = current.super
        end
        if type(addon.OnInitialize) == "function" then
                addon:OnInitialize(name)
        end
        RegisterOnEnable(addon)
end

function AceAddon.prototype:PrintAddonInfo()
        local x
        if self.title then
                x = "|cffffff7f" .. tostring(self.title) .. "|r"
        elseif self.name then
                x = "|cffffff7f" .. tostring(self.name) .. "|r"
        else
                x = "|cffffff7f<" .. tostring(self.class) .. " instance>|r"
        end
        if type(self.IsActive) == "function" then
                if not self:IsActive() then
                        x = x .. " " .. STANDBY
                end
        end
        if self.version then
                x = x .. " - |cffffff7f" .. tostring(self.version) .. "|r"
        end
        if self.notes then
                x = x .. " - " .. tostring(self.notes)
        end
        print(x)
        if self.author then
                print(" - |cffffff7f" .. AUTHOR .. ":|r " .. tostring(self.author))
        end
        if self.date then
                print(" - |cffffff7f" .. DATE .. ":|r " .. tostring(self.date))
        end
        if self.category then
                local category = CATEGORIES[self.category]
                if category then
                        print(" - |cffffff7f" .. CATEGORY .. ":|r " .. category)
                end
        end
        if self.email then
                print(" - |cffffff7f" .. EMAIL .. ":|r " .. tostring(self.email))
        end
        if self.website then
                print(" - |cffffff7f" .. WEBSITE .. ":|r " .. tostring(self.website))
        end
end

local options
function AceAddon:GetAceOptionsDataTable(target)
        if not options then
                options = {
                        about = {
                                name = ABOUT,
                                desc = PRINT_ADDON_INFO,
                                type = "execute",
                                func = "PrintAddonInfo",
                                order = -1,
                        }
                }
        end
        return options
end

function AceAddon:PLAYER_LOGIN()
        self.playerLoginFired = true
        if self.addonsToOnEnable then
                while table.getn(self.addonsToOnEnable) > 0 do
                        local addon = table.remove(self.addonsToOnEnable, 1)
                        self.addonsStarted[addon] = true
                        if (type(addon.IsActive) ~= "function" or addon:IsActive()) and (not AceModuleCore or not AceModuleCore:IsModule(addon) or AceModuleCore:IsModuleActive(addon)) then
                                local current = addon.class
                                while true do
                                        if current == AceOO.Class then
                                                break
                                        end
                                        if current.mixins then
                                                for mixin in pairs(current.mixins) do
                                                        if type(mixin.OnEmbedEnable) == "function" then
                                                                mixin:OnEmbedEnable(addon)
                                                        end
                                                end
                                        end
                                        current = current.super
                                end
                                if type(addon.OnEnable) == "function" then
                                        addon:OnEnable()
                                end
                        end
                end
                self.addonsToOnEnable = nil
        end
end

function AceAddon.prototype:Inject(t)
        AceAddon:argCheck(t, 2, "table")
        for k,v in pairs(t) do
                self[k] = v
        end
end

function AceAddon.prototype:init()
        if not AceEvent then
                error(MAJOR_VERSION .. " requires AceEvent-2.0", 4)
        end
        AceAddon.super.prototype.init(self)
        
        self.super = self.class.prototype
        
        AceAddon:RegisterEvent("ADDON_LOADED", "ADDON_LOADED", true)
        table.insert(AceAddon.nextAddon, self)
end

function AceAddon.prototype:ToString()
        local x
        if type(self.title) == "string" then
                x = self.title
        elseif type(self.name) == "string" then
                x = self.name
        else
                x = "<" .. tostring(self.class) .. " instance>"
        end
        if (type(self.IsActive) == "function" and not self:IsActive()) or (AceModuleCore and AceModuleCore:IsModule(addon) and AceModuleCore:IsModuleActive(addon)) then
                x = x .. " " .. STANDBY
        end
        return x
end

AceAddon.new = function(self, m1, m2, m3, m4, m5, m6, m7, m8, m9, m10, m11, m12, m13, m14, m15, m16, m17, m18, m19, m20)
        local class = AceAddon:pcall(AceOO.Classpool, self, m1, m2, m3, m4, m5, m6, m7, m8, m9, m10, m11, m12, m13, m14, m15, m16, m17, m18, m19, m20)
        return class:new()
end

local function external(self, major, instance)
        if major == "AceEvent-2.0" then
                AceEvent = instance
                
                AceEvent:embed(self)
                
                self:RegisterEvent("PLAYER_LOGIN", "PLAYER_LOGIN", true)
        elseif major == "AceConsole-2.0" then
                AceConsole = instance
                
                local slashCommands = { "/ace2" }
                local _,_,_,enabled,loadable = GetAddOnInfo("Ace")
                if not enabled or not loadable then
                        table.insert(slashCommands, "/ace")
                end
                local function listAddon(addon, depth)
                        if not depth then
                                depth = 0
                        end
                        
                        local s = string.rep("  ", depth) .. " - " .. tostring(addon)
                        if rawget(addon, 'version') then
                                s = s .. " - |cffffff7f" .. tostring(addon.version) .. "|r"
                        end
                        if rawget(addon, 'slashCommand') then
                                s = s .. " |cffffff7f(" .. tostring(addon.slashCommand) .. ")|r"
                        end
                        print(s)
                        if type(rawget(addon, 'modules')) == "table" then
                                local i = 0
                                for k,v in pairs(addon.modules) do
                                        i = i + 1
                                        if i == 6 then
                                                print(string.rep("  ", depth + 1) .. " - more...")
                                                break
                                        else
                                                listAddon(v, depth + 1)
                                        end
                                end
                        end
                end
                local function listNormalAddon(i)
                        local name,_,_,enabled,loadable = GetAddOnInfo(i)
                        if not loadable then
                                enabled = false
                        end
                        if self.addons[name] then
                                local addon = self.addons[name]
                                if not AceCoreAddon or not AceCoreAddon:IsModule(addon) then
                                        listAddon(addon)
                                end
                        else
                                local s = " - " .. tostring(GetAddOnMetadata(i, "Title") or name)
                                local version = GetAddOnMetadata(i, "Version")
                                if version then
                                        if string.find(version, "%$Revision: (%d+) %$") then
                                                version = string.gsub(version, "%$Revision: (%d+) %$", "%1")
                                        elseif string.find(version, "%$Rev: (%d+) %$") then
                                                version = string.gsub(version, "%$Rev: (%d+) %$", "%1")
                                        elseif string.find(version, "%$LastChangedRevision: (%d+) %$") then
                                                version = string.gsub(version, "%$LastChangedRevision: (%d+) %$", "%1")
                                        end
                                        s = s .. " - |cffffff7f" .. version .. "|r"
                                end
                                if not enabled then
                                        s = s .. " |cffff0000(disabled)|r"
                                end
                                if IsAddOnLoadOnDemand(i) then
                                        s = s .. " |cff00ff00[LoD]|r"
                                end
                                print(s)
                        end
                end
                local function mySort(alpha, bravo)
                        return tostring(alpha) < tostring(bravo)
                end
                AceConsole.RegisterChatCommand(self, slashCommands, {
                        desc = "AddOn development framework",
                        name = "Ace2",
                        type = "group",
                        args = {
                                about = {
                                        desc = "Get information about Ace2",
                                        name = "About",
                                        type = "execute",
                                        func = function()
                                                print("|cffffff7fAce2|r - |cffffff7f2.0." .. string.gsub(MINOR_VERSION, "%$Revision: (%d+) %$", "%1") .. "|r - AddOn development framework")
                                                print(" - |cffffff7f" .. AUTHOR .. ":|r Ace Development Team")
                                                print(" - |cffffff7f" .. WEBSITE .. ":|r http://www.wowace.com/")
                                        end
                                },
                                list = {
                                        desc = "List addons",
                                        name = "List",
                                        type = "group",
                                        args = {
                                                ace2 = {
                                                        desc = "List addons using Ace2",
                                                        name = "Ace2",
                                                        type = "execute",
                                                        func = function()
                                                                print("|cffffff7fAddon list:|r")
                                                                local AceCoreAddon = AceLibrary:HasInstance("AceCoreAddon-2.0") and AceLibrary("AceCoreAddon-2.0")
                                                                table.sort(self.addons, mySort)
                                                                for _,v in ipairs(self.addons) do
                                                                        if not AceCoreAddon or not AceCoreAddon:IsModule(v) then
                                                                                listAddon(v)
                                                                        end
                                                                end
                                                        end
                                                },
                                                all = {
                                                        desc = "List all addons",
                                                        name = "All",
                                                        type = "execute",
                                                        func = function()
                                                                print("|cffffff7fAddon list:|r")
                                                                local AceCoreAddon = AceLibrary:HasInstance("AceCoreAddon-2.0") and AceLibrary("AceCoreAddon-2.0")
                                                                local count = GetNumAddOns()
                                                                for i = 1, count do
                                                                        listNormalAddon(i)
                                                                end
                                                        end
                                                },
                                                enabled = {
                                                        desc = "List all enabled addons",
                                                        name = "Enabled",
                                                        type = "execute",
                                                        func = function()
                                                                print("|cffffff7fAddon list:|r")
                                                                local AceCoreAddon = AceLibrary:HasInstance("AceCoreAddon-2.0") and AceLibrary("AceCoreAddon-2.0")
                                                                local count = GetNumAddOns()
                                                                for i = 1, count do
                                                                        local _,_,_,enabled,loadable = GetAddOnInfo(i)
                                                                        if enabled and loadable then
                                                                                listNormalAddon(i)
                                                                        end
                                                                end
                                                        end
                                                },
                                                disabled = {
                                                        desc = "List all disabled addons",
                                                        name = "Disabled",
                                                        type = "execute",
                                                        func = function()
                                                                print("|cffffff7fAddon list:|r")
                                                                local AceCoreAddon = AceLibrary:HasInstance("AceCoreAddon-2.0") and AceLibrary("AceCoreAddon-2.0")
                                                                local count = GetNumAddOns()
                                                                for i = 1, count do
                                                                        local _,_,_,enabled,loadable = GetAddOnInfo(i)
                                                                        if not enabled or not loadable then
                                                                                listNormalAddon(i)
                                                                        end
                                                                end
                                                        end
                                                },
                                                lod = {
                                                        desc = "List all LoadOnDemand addons",
                                                        name = "LoadOnDemand",
                                                        type = "execute",
                                                        func = function()
                                                                print("|cffffff7fAddon list:|r")
                                                                local AceCoreAddon = AceLibrary:HasInstance("AceCoreAddon-2.0") and AceLibrary("AceCoreAddon-2.0")
                                                                local count = GetNumAddOns()
                                                                for i = 1, count do
                                                                        if IsAddOnLoadOnDemand(i) then
                                                                                listNormalAddon(i)
                                                                        end
                                                                end
                                                        end
                                                },
                                                ace1 = {
                                                        desc = "List all addons using Ace1",
                                                        name = "Ace 1.x",
                                                        type = "execute",
                                                        func = function()
                                                                print("|cffffff7fAddon list:|r")
                                                                local count = GetNumAddOns()
                                                                for i = 1, count do
                                                                        local dep1, dep2, dep3, dep4 = GetAddOnDependencies(i)
                                                                        if dep1 == "Ace" or dep2 == "Ace" or dep3 == "Ace" or dep4 == "Ace" then
                                                                                listNormalAddon(i)
                                                                        end
                                                                end
                                                        end
                                                },
                                                libs = {
                                                        desc = "List all libraries using AceLibrary",
                                                        name = "Libraries",
                                                        type = "execute",
                                                        func = function()
                                                                if type(AceLibrary) == "table" and type(AceLibrary.libs) == "table" then
                                                                        print("|cffffff7fLibrary list:|r")
                                                                        for name, data in pairs(AceLibrary.libs) do
                                                                                local s
                                                                                if data.minor then
                                                                                        s = " - " .. tostring(name) .. "." .. tostring(data.minor)
                                                                                else
                                                                                        s = " - " .. tostring(name)
                                                                                end
                                                                                if AceLibrary(name).slashCommand then
                                                                                        s = s .. " |cffffff7f(" .. tostring(AceLibrary(name).slashCommand) .. "|cffffff7f)"
                                                                                end
                                                                                print(s)
                                                                        end
                                                                end
                                                        end
                                                },
                                                search = {
                                                        desc = "Search by name",
                                                        name = "Search",
                                                        type = "text",
                                                        usage = "<keyword>",
                                                        input = true,
                                                        get = false,
                                                        set = function(...)
                                                                for i,v in ipairs(arg) do
                                                                        arg[i] = string.lower(string.gsub(string.gsub(v, '%*', '.*'), '%%', '%%%%'))
                                                                end
                                                                local count = GetNumAddOns()
                                                                for i = 1, count do
                                                                        local name = GetAddOnInfo(i)
                                                                        local good = true
                                                                        for _,v in ipairs(arg) do
                                                                                if not string.find(string.lower(name), v) then
                                                                                        good = false
                                                                                        break
                                                                                end
                                                                        end
                                                                        if good then
                                                                                listNormalAddon(i)
                                                                        end
                                                                end
                                                        end
                                                }
                                        },
                                },
                                enable = {
                                        desc = "Enable addon",
                                        name = "Enable",
                                        type = "text",
                                        usage = "<addon>",
                                        get = false,
                                        set = function(text)
                                                local name,title,_,_,_,reason = GetAddOnInfo(text)
                                                if reason == "MISSING" then
                                                        print(string.format("|cffffff7fAce2:|r AddOn %q does not exist", text))
                                                else
                                                        EnableAddOn(text)
                                                        print(string.format("|cffffff7fAce2:|r %s is now enabled", title or name))
                                                end
                                        end,
                                },
                                disable = {
                                        desc = "Disable addon",
                                        name = "Disable",
                                        type = "text",
                                        usage = "<addon>",
                                        get = false,
                                        set = function(text)
                                                local name,title,_,_,_,reason = GetAddOnInfo(text)
                                                if reason == "MISSING" then
                                                        print(string.format("|cffffff7fAce2:|r AddOn %q does not exist", text))
                                                else
                                                        DisableAddOn(text)
                                                        print(string.format("|cffffff7fAce2:|r %s is now disabled", title or name))
                                                end
                                        end,
                                },
                                load = {
                                        desc = "Load addon",
                                        name = "Load",
                                        type = "text",
                                        usage = "<addon>",
                                        get = false,
                                        set = function(text)
                                                local name,title,_,_,loadable,reason = GetAddOnInfo(text)
                                                if reason == "MISSING" then
                                                        print(string.format("|cffffff7fAce2:|r AddOn %q does not exist.", text))
                                                elseif not loadable then
                                                        print(string.format("|cffffff7fAce2:|r AddOn %q is not loadable. Reason: %s", text, reason))
                                                else
                                                        LoadAddOn(text)
                                                        print(string.format("|cffffff7fAce2:|r %s is now loaded", title or name))
                                                end
                                        end
                                },
                                info = {
                                        desc = "Display information",
                                        name = "Information",
                                        type = "execute",
                                        func = function()
                                                local mem, threshold = gcinfo()
                                                print(string.format(" - |cffffff7fMemory usage [|r%.3f MiB|cffffff7f]|r", mem / 1024))
                                                print(string.format(" - |cffffff7fThreshold [|r%.3f MiB|cffffff7f]|r", threshold / 1024))
                                                print(string.format(" - |cffffff7fFramerate [|r%.0f fps|cffffff7f]|r", GetFramerate()))
                                                local bandwidthIn, bandwidthOut, latency = GetNetStats()
                                                bandwidthIn, bandwidthOut = floor(bandwidthIn * 1024), floor(bandwidthOut * 1024)
                                                print(string.format(" - |cffffff7fLatency [|r%.0f ms|cffffff7f]|r", latency))
                                                print(string.format(" - |cffffff7fBandwidth in [|r%.0f B/s|cffffff7f]|r", bandwidthIn))
                                                print(string.format(" - |cffffff7fBandwidth out [|r%.0f B/s|cffffff7f]|r", bandwidthOut))
                                                print(string.format(" - |cffffff7fTotal addons [|r%d|cffffff7f]|r", GetNumAddOns()))
                                                print(string.format(" - |cffffff7fAce2 addons [|r%d|cffffff7f]|r", table.getn(self.addons)))
                                                local ace = 0
                                                local enabled = 0
                                                local disabled = 0
                                                local lod = 0
                                                for i = 1, GetNumAddOns() do
                                                        local dep1, dep2, dep3, dep4 = GetAddOnDependencies(i)
                                                        if dep1 == "Ace" or dep2 == "Ace" or dep3 == "Ace" or dep4 == "Ace" then
                                                                ace = ace + 1
                                                        end
                                                        if IsAddOnLoadOnDemand(i) then
                                                                lod = lod + 1
                                                        end
                                                        local _,_,_,IsActive,loadable = GetAddOnInfo(i)
                                                        if not IsActive or not loadable then
                                                                disabled = disabled + 1
                                                        else
                                                                enabled = enabled + 1
                                                        end
                                                end
                                                print(string.format(" - |cffffff7fAce 1.x addons [|r%d|cffffff7f]|r", ace))
                                                print(string.format(" - |cffffff7fLoadOnDemand addons [|r%d|cffffff7f]|r", lod))
                                                print(string.format(" - |cffffff7fenabled addons [|r%d|cffffff7f]|r", enabled))
                                                print(string.format(" - |cffffff7fdisabled addons [|r%d|cffffff7f]|r", disabled))
                                                local libs = 0
                                                if type(AceLibrary) == "table" and type(AceLibrary.libs) == "table" then
                                                        for _ in pairs(AceLibrary.libs) do
                                                                libs = libs + 1
                                                        end
                                                end
                                                print(string.format(" - |cffffff7fAceLibrary instances [|r%d|cffffff7f]|r", libs))
                                        end
                                }
                        }
                })
        elseif major == "AceModuleCore-2.0" then
                AceModuleCore = instance
        end
end

local function activate(self, oldLib, oldDeactivate)
        AceAddon = self
        
        if oldLib then
                self.playerLoginFired = oldLib.playerLoginFired or DEFAULT_CHAT_FRAME and DEFAULT_CHAT_FRAME.defaultLanguage
                self.addonsToOnEnable = oldLib.addonsToOnEnable
                self.addons = oldLib.addons
                self.nextAddon = oldLib.nextAddon
                self.addonsStarted = oldLib.addonsStarted
        end
        if not self.addons then
                self.addons = {}
        end
        if not self.nextAddon then
                self.nextAddon = {}
        end
        if not self.addonsStarted then
                self.addonsStarted = {}
        end
        if oldDeactivate then
                oldDeactivate(oldLib)
        end
end

AceLibrary:Register(AceAddon, MAJOR_VERSION, MINOR_VERSION, activate, nil, external)
AceAddon = AceLibrary(MAJOR_VERSION)