Quantcast

- Fixed missing elements from the Titan folder.

HonorGoG [10-06-18 - 00:17]
- Fixed missing elements from the Titan folder.
- Updated toc.
Filename
Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom0.tga
Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom1.tga
Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop0.tga
Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop1.tga
Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom0.blp
Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom1.blp
Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop0.blp
Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop1.blp
Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom0.tga
Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom1.tga
Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop0.tga
Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop1.tga
Titan/Artwork/TitanPanelBackgroundBottom0.tga
Titan/Artwork/TitanPanelBackgroundBottom1.tga
Titan/Artwork/TitanPanelBackgroundTop0.tga
Titan/Artwork/TitanPanelBackgroundTop1.tga
Titan/Artwork/TitanPanelPushpinIn.tga
Titan/Artwork/TitanPanelPushpinOut.tga
Titan/LDBToTitan.lua
Titan/LibDataBroker-1.1.lua
Titan/TitanAutoHide.lua
Titan/TitanAutoHide.xml
Titan/TitanConfig.lua
Titan/TitanPanel.lua
Titan/TitanPanel.xml
Titan/TitanPanelChangeLog.txt
Titan/TitanPanelSetup-READ_ME_FIRST.txt
Titan/TitanPanelTemplate.lua
Titan/TitanPanelTemplate.xml
Titan/TitanPluginDoc.lua
Titan/TitanUtils.lua
Titan/TitanVariables.lua
Titan/embeds.xml
Titan/libs/!LibUIDropDownMenu/!LibUIDropDownMenu.toc
Titan/libs/!LibUIDropDownMenu/Docs/Readme.txt
Titan/libs/!LibUIDropDownMenu/Docs/Revision.txt
Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.lua
Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.toc
Titan/libs/!LibUIDropDownMenu/LibStub/tests/test.lua
Titan/libs/!LibUIDropDownMenu/LibStub/tests/test2.lua
Titan/libs/!LibUIDropDownMenu/LibStub/tests/test3.lua
Titan/libs/!LibUIDropDownMenu/LibStub/tests/test4.lua
Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibEasyMenu.lua
Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.lua
Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.xml
Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.lua
Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.xml
Titan/libs/Ace3.lua
Titan/libs/Ace3.toc
Titan/libs/AceAddon-3.0/AceAddon-3.0.lua
Titan/libs/AceAddon-3.0/AceAddon-3.0.xml
Titan/libs/AceConfig-3.0/AceConfig-3.0.lua
Titan/libs/AceConfig-3.0/AceConfig-3.0.xml
Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.lua
Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.xml
Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.lua
Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.xml
Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.lua
Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.xml
Titan/libs/AceGUI-3.0-SharedMediaWidgets.toc
Titan/libs/AceGUI-3.0-SharedMediaWidgets/BackgroundWidget.lua
Titan/libs/AceGUI-3.0-SharedMediaWidgets/BorderWidget.lua
Titan/libs/AceGUI-3.0-SharedMediaWidgets/FontWidget.lua
Titan/libs/AceGUI-3.0-SharedMediaWidgets/SoundWidget.lua
Titan/libs/AceGUI-3.0-SharedMediaWidgets/StatusbarWidget.lua
Titan/libs/AceGUI-3.0-SharedMediaWidgets/prototypes.lua
Titan/libs/AceGUI-3.0-SharedMediaWidgets/widget.xml
Titan/libs/AceGUI-3.0/AceGUI-3.0.lua
Titan/libs/AceGUI-3.0/AceGUI-3.0.xml
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua
Titan/libs/AceHook-3.0/AceHook-3.0.lua
Titan/libs/AceHook-3.0/AceHook-3.0.xml
Titan/libs/AceLocale-3.0/AceLocale-3.0.lua
Titan/libs/AceLocale-3.0/AceLocale-3.0.xml
Titan/libs/AceTimer-3.0/AceTimer-3.0.lua
Titan/libs/AceTimer-3.0/AceTimer-3.0.xml
Titan/libs/Bindings.xml
Titan/libs/CHANGES.txt
Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.lua
Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.xml
Titan/libs/Changelog-Ace3-Release-r1151.txt
Titan/libs/Changelog-AceGUI-3.0-SharedMediaWidgets-r57-release.txt
Titan/libs/LICENSE.txt
Titan/libs/LibQTip-1.0/CHANGES.txt
Titan/libs/LibQTip-1.0/LICENSE.txt
Titan/libs/LibQTip-1.0/LibQTip-1.0.lua
Titan/libs/LibQTip-1.0/LibQTip-1.0.toc
Titan/libs/LibQTip-1.0/LibStub/LibStub.lua
Titan/libs/LibQTip-1.0/lib.xml
Titan/libs/LibSharedMedia-3.0/CHANGES.txt
Titan/libs/LibSharedMedia-3.0/Changelog-LibSharedMedia-3.0-7.0.3.txt
Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua
Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.toc
Titan/libs/LibSharedMedia-3.0/lib.xml
Titan/libs/LibStub/LibStub.lua
Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.lua
Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.xml
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua
Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.lua
Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.xml
Titan/libs/Libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua
Titan/libs/Libs/LibSharedMedia-3.0/lib.xml
Titan/libs/Libs/LibStub/LibStub.lua
Titan/libs/Libs/LibStub/LibStub.toc
Titan/libs/changelog.txt
Titan/libs/lib.xml
Titan/libs/widget.xml
Titan/license.txt
Titan/locale/Localization.BR.lua
Titan/locale/Localization.CN.lua
Titan/locale/Localization.DE.lua
Titan/locale/Localization.ES.lua
Titan/locale/Localization.FR.lua
Titan/locale/Localization.IT.lua
Titan/locale/Localization.KR.lua
Titan/locale/Localization.RU.lua
Titan/locale/Localization.TW.lua
Titan/locale/Localization.lua
diff --git a/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom0.tga b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom0.tga
new file mode 100644
index 0000000..e56e01f
Binary files /dev/null and b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom0.tga differ
diff --git a/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom1.tga b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom1.tga
new file mode 100644
index 0000000..e56e01f
Binary files /dev/null and b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundBottom1.tga differ
diff --git a/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop0.tga b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop0.tga
new file mode 100644
index 0000000..e56e01f
Binary files /dev/null and b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop0.tga differ
diff --git a/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop1.tga b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop1.tga
new file mode 100644
index 0000000..e56e01f
Binary files /dev/null and b/Titan/Artwork/Custom/BlackPlusOne Skin/TitanPanelBackgroundTop1.tga differ
diff --git a/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..36935e9
Binary files /dev/null and b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..c0c6d28
Binary files /dev/null and b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..8cdf073
Binary files /dev/null and b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..8cdf073
Binary files /dev/null and b/Titan/Artwork/Custom/Charcoal Metal/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..e9b29e0
Binary files /dev/null and b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..f2304d2
Binary files /dev/null and b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..47c6aae
Binary files /dev/null and b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..058f5b6
Binary files /dev/null and b/Titan/Artwork/Custom/Christmas Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..e829388
Binary files /dev/null and b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..261a05a
Binary files /dev/null and b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..0b9be6b
Binary files /dev/null and b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..d623075
Binary files /dev/null and b/Titan/Artwork/Custom/Crusader Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..7e73492
Binary files /dev/null and b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..8b7f79c
Binary files /dev/null and b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..8b72475
Binary files /dev/null and b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..cfcf3cf
Binary files /dev/null and b/Titan/Artwork/Custom/Cursed Orange Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..854205b
Binary files /dev/null and b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..a29b07c
Binary files /dev/null and b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..a6b01ce
Binary files /dev/null and b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..c6c2461
Binary files /dev/null and b/Titan/Artwork/Custom/Dark Wood Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..66d3f93
Binary files /dev/null and b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..ed40f5c
Binary files /dev/null and b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..7032308
Binary files /dev/null and b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..0600d95
Binary files /dev/null and b/Titan/Artwork/Custom/Deep Cave Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..bea8466
Binary files /dev/null and b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..3a91710
Binary files /dev/null and b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..3075011
Binary files /dev/null and b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..84c6755
Binary files /dev/null and b/Titan/Artwork/Custom/Elfwood Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..33dec72
Binary files /dev/null and b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..b27b3cf
Binary files /dev/null and b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..c0cf566
Binary files /dev/null and b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..49c0074
Binary files /dev/null and b/Titan/Artwork/Custom/Engineer Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..6e767ae
Binary files /dev/null and b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..6e767ae
Binary files /dev/null and b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..64cda10
Binary files /dev/null and b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..64cda10
Binary files /dev/null and b/Titan/Artwork/Custom/Frozen Metal Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..bb402a4
Binary files /dev/null and b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..ed19072
Binary files /dev/null and b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..44a70cd
Binary files /dev/null and b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..44a70cd
Binary files /dev/null and b/Titan/Artwork/Custom/Graphic Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..28d466c
Binary files /dev/null and b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..e0a82c9
Binary files /dev/null and b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..5def9df
Binary files /dev/null and b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..5b49166
Binary files /dev/null and b/Titan/Artwork/Custom/Graveyard Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..bbeee6c
Binary files /dev/null and b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..74f8c69
Binary files /dev/null and b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..e90928b
Binary files /dev/null and b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..d5bc403
Binary files /dev/null and b/Titan/Artwork/Custom/Hidden Leaf Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..062cc86
Binary files /dev/null and b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..882b24d
Binary files /dev/null and b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..317b19c
Binary files /dev/null and b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..ac6c11e
Binary files /dev/null and b/Titan/Artwork/Custom/Holy Warrior Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..c454816
Binary files /dev/null and b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..c454816
Binary files /dev/null and b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..dafc5b9
Binary files /dev/null and b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..dafc5b9
Binary files /dev/null and b/Titan/Artwork/Custom/Nightlife Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..1dbd937
Binary files /dev/null and b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..e4b14e1
Binary files /dev/null and b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..22cc85c
Binary files /dev/null and b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..700f567
Binary files /dev/null and b/Titan/Artwork/Custom/Orgrimmar Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..efb5404
Binary files /dev/null and b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..2ff44bc
Binary files /dev/null and b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..73a8ba7
Binary files /dev/null and b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..7525546
Binary files /dev/null and b/Titan/Artwork/Custom/Plate Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom0.blp b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom0.blp
new file mode 100644
index 0000000..9915d0e
Binary files /dev/null and b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom0.blp differ
diff --git a/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom1.blp b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom1.blp
new file mode 100644
index 0000000..9915d0e
Binary files /dev/null and b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundBottom1.blp differ
diff --git a/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop0.blp b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop0.blp
new file mode 100644
index 0000000..45be753
Binary files /dev/null and b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop0.blp differ
diff --git a/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop1.blp b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop1.blp
new file mode 100644
index 0000000..9e75c6a
Binary files /dev/null and b/Titan/Artwork/Custom/Tribal Skin/TitanPanelBackgroundTop1.blp differ
diff --git a/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom0.tga b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom0.tga
new file mode 100644
index 0000000..3a1d3fa
Binary files /dev/null and b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom0.tga differ
diff --git a/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom1.tga b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom1.tga
new file mode 100644
index 0000000..3a1d3fa
Binary files /dev/null and b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundBottom1.tga differ
diff --git a/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop0.tga b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop0.tga
new file mode 100644
index 0000000..84ebd84
Binary files /dev/null and b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop0.tga differ
diff --git a/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop1.tga b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop1.tga
new file mode 100644
index 0000000..84ebd84
Binary files /dev/null and b/Titan/Artwork/Custom/X-Perl/TitanPanelBackgroundTop1.tga differ
diff --git a/Titan/Artwork/TitanPanelBackgroundBottom0.tga b/Titan/Artwork/TitanPanelBackgroundBottom0.tga
new file mode 100644
index 0000000..c213c03
Binary files /dev/null and b/Titan/Artwork/TitanPanelBackgroundBottom0.tga differ
diff --git a/Titan/Artwork/TitanPanelBackgroundBottom1.tga b/Titan/Artwork/TitanPanelBackgroundBottom1.tga
new file mode 100644
index 0000000..cf10acb
Binary files /dev/null and b/Titan/Artwork/TitanPanelBackgroundBottom1.tga differ
diff --git a/Titan/Artwork/TitanPanelBackgroundTop0.tga b/Titan/Artwork/TitanPanelBackgroundTop0.tga
new file mode 100644
index 0000000..a3fc230
Binary files /dev/null and b/Titan/Artwork/TitanPanelBackgroundTop0.tga differ
diff --git a/Titan/Artwork/TitanPanelBackgroundTop1.tga b/Titan/Artwork/TitanPanelBackgroundTop1.tga
new file mode 100644
index 0000000..4035c23
Binary files /dev/null and b/Titan/Artwork/TitanPanelBackgroundTop1.tga differ
diff --git a/Titan/Artwork/TitanPanelPushpinIn.tga b/Titan/Artwork/TitanPanelPushpinIn.tga
new file mode 100644
index 0000000..406ee98
Binary files /dev/null and b/Titan/Artwork/TitanPanelPushpinIn.tga differ
diff --git a/Titan/Artwork/TitanPanelPushpinOut.tga b/Titan/Artwork/TitanPanelPushpinOut.tga
new file mode 100644
index 0000000..d3c892b
Binary files /dev/null and b/Titan/Artwork/TitanPanelPushpinOut.tga differ
diff --git a/Titan/LDBToTitan.lua b/Titan/LDBToTitan.lua
new file mode 100644
index 0000000..df164ab
--- /dev/null
+++ b/Titan/LDBToTitan.lua
@@ -0,0 +1,828 @@
+--[[ File
+NAME: LibDataBrokerToTitan.lua
+DESC: A "bridge" module to ensure proper registration and communication of LDB plugins with Titan Panel
+--]]
+--[[ API
+NAME: Titan LDB overview
+DESC: Titan will automatically convert a LDB type addon to a Titan plugin.
+Only LDB types listed in the LDB 1.1 spec are supported. Custom types are not supported.
+
+Supported
+- "launcher" become "icon" plugins
+	icon* - always shown
+	OnClick* -
+	label^ -
+	right side^ - default
+	tooltip
+- "data source" become "combo" plugins with icon; a tooltip/Click; and optional label
+	icon^ -
+	OnClick -
+	text*^ - or value & suffix
+	label^ -
+	OnEnter -
+	OnLeave -
+	tooltip
+	OnTooltipShow -
+
+* required by LDB spec
+^ user controlled show / hide
+:DESC
+--]]
+
+--[[ doc
+-----------------------------------------------------------------
+-- By Titan Dev team
+-- Originally by Tristanian aka "TristTitan" as a Titan member
+-- Created and initially commited on : July 29th, 2008
+-----------------------------------------------------------------
+--]]
+local xcategories = {
+-- Titan categories mapping to match addon metadata information
+	["Combat"] = "Combat",
+	["General"] = "General",
+	["Information"] = "Information",
+	["Interface"] = "Interface",
+	["Profession"] = "Profession",
+-- Ace2 table mapping to Titan categories in order to match
+-- addon metadata information
+	["Action Bars"] = "Interface",
+	["Auction"] = "Information",
+	["Audio"] = "Interface",
+	["Battlegrounds/PvP"] = "Information",
+	["Buffs"] = "Information",
+	["Chat/Communication"] = "Interface",
+	["Druid"] = "Information",
+	["Hunter"] = "Information",
+	["Mage"] = "Information",
+	["Paladin"] = "Information",
+	["Priest"] = "Information",
+	["Rogue"] = "Information",
+	["Shaman"] = "Information",
+	["Warlock"] = "Information",
+	["Warrior"] = "Information",
+	["Healer"] = "Information",
+	["Tank"] = "Information",
+	["Caster"] = "Information",
+--	["Combat"] = "Combat",
+	["Compilations"] = "General",
+	["Data Export"] = "General",
+	["Development Tools "] = "General",
+	["Guild"] = "Information",
+	["Frame Modification"] = "Interface",
+	["Interface Enhancements"] = "Interface",
+	["Inventory"] = "Information",
+	["Library"] = "General",
+	["Map"] = "Interface",
+	["Mail"] = "Information",
+	["Miscellaneous"] = "General",
+	["Misc"] = "General",
+	["Quest"] = "Information",
+	["Raid"] = "Information",
+	["Tradeskill"] = "Profession",
+	["UnitFrame"] = "Interface",
+}
+local LAUNCHER = "launcher"
+local DATA_SOURCE = "data source"
+local SupportedDOTypes = {DATA_SOURCE, LAUNCHER} -- in the 1.1 spec
+
+-- constants & variables
+local CALLBACK_PREFIX = "LibDataBroker_AttributeChanged_"
+local NAME_PREFIX = "" --"LDBT_"
+local _G = getfenv(0);
+local InCombatLockdown	= _G.InCombatLockdown;
+-- Create control frame so we can get events
+local LDBToTitan = CreateFrame("Frame", "LDBTitan")
+local ldb = LibStub:GetLibrary("LibDataBroker-1.1")
+local Tablet, LibQTip = nil, nil
+local media = LibStub("LibSharedMedia-3.0")
+-- generic icon in case the DO does not provide one
+local iconTitanDefault = "Interface\\PVPFrame\\\PVP-ArenaPoints-Icon";
+
+-- Events we want for LDBToTitan
+LDBToTitan:RegisterEvent("PLAYER_LOGIN")
+LDBToTitan:RegisterEvent("PLAYER_ENTERING_WORLD")
+
+--[[ Titan
+NAME: LDBToTitan:TitanLDBSetOwnerPosition
+DESC: Properly anchor tooltips of the Titan (LDB) plugin
+VAR: parent
+VAR: anchorPoint
+VAR: relativeToFrame
+VAR: relativePoint
+VAR: xOffset
+VAR: yOffset
+VAR: frame
+--]]
+function LDBToTitan:TitanLDBSetOwnerPosition(parent, anchorPoint, relativeToFrame, relativePoint, xOffset, yOffset, frame)
+	if frame:GetName() == "GameTooltip" then
+		frame:SetOwner(parent, "ANCHOR_NONE");
+		-- set alpha (transparency) for the Game Tooltip
+		local red, green, blue = GameTooltip:GetBackdropColor();
+		local red2, green2, blue2 = GameTooltip:GetBackdropBorderColor();
+		frame:SetBackdropColor(red,green,blue,TitanPanelGetVar("TooltipTrans"));
+		frame:SetBackdropBorderColor(red2,green2,blue2,
+			TitanPanelGetVar("TooltipTrans"));
+
+		-- set font size for the Game Tooltip
+		if not TitanPanelGetVar("DisableTooltipFont") then
+			if TitanTooltipScaleSet < 1 then
+				TitanTooltipOrigScale = GameTooltip:GetScale();
+				TitanTooltipScaleSet = TitanTooltipScaleSet + 1;
+			end
+			frame:SetScale(TitanPanelGetVar("TooltipFont"));
+		end
+	end
+	frame:ClearAllPoints();
+	frame:SetPoint(anchorPoint, relativeToFrame, relativePoint, xOffset, yOffset);
+end
+
+--[[ Titan
+NAME: LDBToTitan:TitanLDBSetTooltip
+DESC: Fill in the tooltip for the Titan (LDB) plugin
+VAR: name - Titan id of the plugin
+VAR: frame - tooltip frame
+VAR: func - tooltip function to be run
+OUT:  None
+--]]
+function LDBToTitan:TitanLDBSetTooltip(name, frame, func)
+-- Check to see if we allow tooltips to be shown
+	if not TitanPanelGetVar("ToolTipsShown")
+	or (TitanPanelGetVar("HideTipsInCombat") and InCombatLockdown()) then
+		return
+	end
+
+	local button = TitanUtils_GetButton(name);
+	local scale = TitanPanelGetVar("Scale");
+	local offscreenX, offscreenY;
+	local i = TitanPanel_GetButtonNumber(name);
+	local bar = TITAN_PANEL_DISPLAY_PREFIX..TitanUtils_GetWhichBar(name)
+	local vert = TitanBarData[bar].vert
+	-- Get TOP or BOTTOM for the anchor and relative anchor
+	local rel_pt, pt
+	if vert == TITAN_TOP then
+		pt = "TOP"
+		rel_pt = "BOTTOM"
+	else
+		pt = "BOTTOM"
+		rel_pt = "TOP"
+	end
+
+	if _G[bar] then
+		self:TitanLDBSetOwnerPosition(button, pt.."LEFT", button:GetName(),
+			rel_pt.."LEFT", -10, 0, frame);	-- y 4 * scale
+		-- Adjust frame position if it's off the screen
+		offscreenX, offscreenY = TitanUtils_GetOffscreen(frame);
+		if ( offscreenX == -1 ) then
+			self:TitanLDBSetOwnerPosition(button, pt.."LEFT", bar,
+				rel_pt.."LEFT", 0, 0, frame);
+		elseif ( offscreenX == 1 ) then
+			self:TitanLDBSetOwnerPosition(button, pt.."RIGHT", bar,
+				rel_pt.."RIGHT", 0, 0, frame);
+		end
+	else
+	end
+
+	if func then func(frame) end; -- TODO: use pcall??
+	frame:Show();
+end
+
+--[[ Titan
+NAME: LDBToTitan:TitanLDBHandleScripts
+DESC: Script Handler for the Titan (LDB) plugin
+VAR: event - event to process
+VAR: name -  id of the plugin
+VAR: _ - not used
+VAR: func - function to be run
+VAR: obj - LDB object
+OUT:  None
+NOTE:
+- This implementation will work fine for a static tooltip but may have implications for dynamic ones so for now, we'll only set it once (no callback) and see what happens
+:NOTE
+--]]
+function LDBToTitan:TitanLDBHandleScripts(event, name, _, func, obj)
+	local TitanPluginframe = _G["TitanPanel"..NAME_PREFIX..name.."Button"];
+
+	-- tooltip
+	if event:find("tooltip") and not event:find("OnTooltipShow") then
+		local pluginframe = _G[obj.tooltip] or obj.tooltip
+		if pluginframe then
+			TitanPluginframe:SetScript("OnEnter", function(self)
+				TitanPanelButton_OnEnter(self);
+				LDBToTitan:TitanLDBSetTooltip(NAME_PREFIX..name, pluginframe, nil)
+				end
+				)
+
+			TitanPluginframe:SetScript("OnMouseDown", function(self)
+				pluginframe:Hide();
+				end
+				)
+
+			if pluginframe:GetScript("OnLeave") then
+				-- do nothing
+			else
+				TitanPluginframe:SetScript("OnLeave", function(self)
+					if obj.OnLeave then
+						obj.OnLeave(self)
+					end
+					pluginframe:Hide();
+					TitanPanelButton_OnLeave(self);
+					end
+					)
+			end
+
+			if pluginframe:GetName()~="GameTooltip" then
+				if pluginframe:GetScript("OnShow") then
+					-- do nothing
+				else
+					pluginframe:SetScript("OnShow", function(self)
+						LDBToTitan:TitanLDBSetTooltip(NAME_PREFIX..name, pluginframe, nil)
+						end
+						)
+				end
+			end
+		end
+
+	-- OnTooltipShow
+	elseif event:find("OnTooltipShow") then
+		TitanPluginframe:SetScript("OnEnter", function(self)
+			if TITAN_PANEL_MOVING == 0 and func then
+				LDBToTitan:TitanLDBSetTooltip(NAME_PREFIX..name, GameTooltip, func);
+			end
+				TitanPanelButton_OnEnter(self);
+			end
+			)
+		TitanPluginframe:SetScript("OnLeave", function(self)
+			GameTooltip:Hide();
+			TitanPanelButton_OnLeave(self);
+			end
+			)
+
+	-- OnDoubleClick
+	elseif event:find("OnDoubleClick") and not event:find("OnClick") then
+		TitanPluginframe:SetScript("OnDoubleClick", function(self, button)
+			if TITAN_PANEL_MOVING == 0 then
+				func(self, button)
+			end
+			end
+			)
+
+	-- OnClick
+	elseif event:find("OnClick") then
+		TitanPluginframe:SetScript("OnClick", function(self, button)
+			if TITAN_PANEL_MOVING == 0 then
+				func(self, button)
+			end
+			-- implement a safeguard, since the DO may actually use
+			-- Blizzy dropdowns !
+			if not TitanPanelRightClickMenu_IsVisible() then
+				TitanPanelButton_OnClick(self, button);
+			else
+				TitanUtils_CloseAllControlFrames();
+			end
+			end
+			)
+	-- OnEnter
+	else
+		TitanPluginframe:SetScript("OnEnter", function(self)
+			-- Check for tooltip libs without embedding them
+			if AceLibrary and AceLibrary:HasInstance("Tablet-2.0") then
+				Tablet = AceLibrary("Tablet-2.0")
+			end
+			LibQTip = LibStub("LibQTip-1.0", true)
+			-- Check to see if we allow tooltips to be shown
+			if not TitanPanelGetVar("ToolTipsShown")
+			or (TitanPanelGetVar("HideTipsInCombat") and InCombatLockdown()) then
+				-- if a plugin is using tablet, then detach and close the tooltip
+				if Tablet and Tablet:IsRegistered(TitanPluginframe)
+				and Tablet:IsAttached(TitanPluginframe) then
+					Tablet:Detach(TitanPluginframe);
+					Tablet:Close(TitanPluginframe);
+				end
+				return;
+			else
+				-- if a plugin is using tablet, then re-attach the tooltip
+				-- (it will auto-open on mouseover)
+				if Tablet and Tablet:IsRegistered(TitanPluginframe)
+				and not Tablet:IsAttached(TitanPluginframe) then
+					Tablet:Attach(TitanPluginframe);
+				end
+			end
+			-- if a plugin is using tablet then set its transparency
+			-- and font size accordingly
+			if Tablet and Tablet:IsRegistered(TitanPluginframe) then
+				Tablet:SetTransparency(TitanPluginframe, TitanPanelGetVar("TooltipTrans"))
+				if not TitanPanelGetVar("DisableTooltipFont") then
+					Tablet:SetFontSizePercent(TitanPluginframe, TitanPanelGetVar("TooltipFont"))
+				elseif TitanPanelGetVar("DisableTooltipFont")
+				and Tablet:GetFontSizePercent(TitanPluginframe)~=1 then
+					Tablet:SetFontSizePercent(TitanPluginframe, 1)
+				end
+			end
+			-- set original tooltip scale for GameTooltip
+			if not TitanPanelGetVar("DisableTooltipFont") then
+				TitanTooltipOrigScale = GameTooltip:GetScale();
+			end
+			-- call OnEnter on LDB Object
+			if TITAN_PANEL_MOVING == 0 and func then
+				func(self)
+			end
+			TitanPanelButton_OnEnter(self);
+			-- LibQTip-1.0 support code
+			if LibQTip then
+				local tt = nil
+				local key, tip
+				for key, tip in LibQTip:IterateTooltips() do
+					if tip then
+						local _, relativeTo = tip:GetPoint()
+						if relativeTo
+						and relativeTo:GetName() == TitanPluginframe:GetName() then
+							tt = tip
+							break
+						end
+					end
+				end
+				if tt then
+					-- set transparency
+					local red, green, blue, _ = tt:GetBackdropColor()
+					local red2, green2, blue2, _ = tt:GetBackdropBorderColor()
+					tt:SetBackdropColor(red,green,blue,
+						TitanPanelGetVar("TooltipTrans"))
+					tt:SetBackdropBorderColor(red2,green2,blue2,
+						TitanPanelGetVar("TooltipTrans"))
+				end
+			end
+			-- /LibQTip-1.0 support code
+			end
+			)
+
+	-- OnLeave
+		TitanPluginframe:SetScript("OnLeave", function(self)
+			if obj.OnLeave then
+				obj.OnLeave(self)
+			end
+			TitanPanelButton_OnLeave(self);
+			end
+			)
+	end
+end
+
+--[[ Titan
+NAME: LDBToTitan:TitanLDBTextUpdate
+DESC: Text callback for the Titan (LDB) plugin when the LDB addon changes display text of the LDB object
+VAR: _ - not used
+VAR: name -  id of the plugin
+VAR: event - event to process
+VAR: attr - LDB obj attribute (field) that changed
+VAR: value - new value of attr
+VAR: dataobj - LDB object
+OUT:  None
+--]]
+function LDBToTitan:TitanLDBTextUpdate(_, name,  attr, value, dataobj)
+	-- just in case the LDB is active before Titan can register it...
+	if not Titan__InitializedPEW then
+		-- plugins have not been registered yet.
+		return
+	end
+	-- This check is overkill but just in case...
+	local plugin = TitanUtils_GetPlugin(NAME_PREFIX..name)
+	local ldb = plugin and plugin.LDBVariables
+	if not ldb then
+		-- This plugin has not been registered
+		return
+	end
+
+	-- Accept the various display elements and update the Titan plugin
+	if attr == "value" then ldb.value = value end
+	if attr == "suffix" then ldb.suffix = value end
+	if attr == "text" then ldb.text = value end
+	if attr == "label" then ldb.label = value end
+
+	-- Now update the button with the change
+	TitanPanelButton_UpdateButton(NAME_PREFIX..name)
+end
+
+--[[ Titan
+NAME: TitanLDBShowText
+DESC: Text callback for the Titan (LDB) plugin when the LDB addon changes display text
+VAR: name -  id of the plugin
+OUT: None
+NOTE:
+- One interpretation of 1.1 spec to show text is either
+1) use .text or use .value & .suffix (Titan implements)
+2) always use .text but .value & .suffix are parts if needed
+:NOTE
+--]]
+function TitanLDBShowText(name)
+	-- Set 'label1' and 'value1' for the Titan button display
+	local nametrim = string.gsub (name, "LDBT_", "");
+	local fontstring = _G["TitanPanel"..NAME_PREFIX..nametrim..TITAN_PANEL_BUTTON_TEXT];
+	local separator = ": "
+	local lab1, val1 = "", ""
+	local plugin = TitanUtils_GetPlugin(name)
+	local ldb = plugin and plugin.LDBVariables
+
+	if ldb then -- sanity check
+		-- Check for display label
+		if TitanGetVar(name, "ShowLabelText") then
+			lab1 = (ldb.label or "")
+		else
+			lab1 = ""
+		end
+
+		if lab1 == "" then
+			-- leave alone
+		else
+			lab1 = lab1..separator
+		end
+
+		-- Check for display text
+		if TitanGetVar(name, "ShowRegularText") then
+			if ldb.suffix and ldb.suffix ~="" then
+				val1 = (ldb.value or "").." "..ldb.suffix
+			else
+				val1 = (ldb.text or "")
+			end
+		end
+	else
+		-- return values will be empty strings
+	end
+
+	if lab1 == "" then
+		lab1 = nil
+	else
+		lab1 = TitanUtils_GetNormalText(lab1)
+	end
+	if val1 == "" then
+		val1 = nil
+	else
+		val1 = TitanGetVar(name, "ShowColoredText")
+			and TitanUtils_GetGreenText(val1) or TitanUtils_GetHighlightText(val1)
+	end
+	return lab1, val1
+end
+
+--[[ Titan
+NAME: LDBToTitan:TitanLDBTextUpdate
+DESC: Icon callback for the Titan (LDB) plugin when the LDB addon changes the icon of the LDB object
+VAR: _ - not used
+VAR: name -  id of the plugin
+VAR: attr - LDB obj attribute (field) that changed
+VAR: value - new value of attr
+VAR: dataobj - LDB object
+OUT: None
+--]]
+function LDBToTitan:TitanLDBIconUpdate(_, name,  attr, value, dataobj)
+	-- just in case the LDB is active before Titan can register it...
+	if not Titan__InitializedPEW then
+		-- no plugins are registered yet
+		return
+	end
+	-- This check is overkill but just in case...
+	local plugin = TitanUtils_GetPlugin(NAME_PREFIX..name)
+	local ldb = plugin and plugin.LDBVariables
+	if not ldb then
+		-- This plugin is not registered yet
+		return
+	end
+
+	if attr == "icon" then
+		TitanPlugins[NAME_PREFIX..name].icon = value;
+		TitanPanelButton_SetButtonIcon(NAME_PREFIX..name);
+	end
+
+	-- support for iconCoords, iconR, iconG, iconB attributes
+	if attr == "iconCoords" then
+		TitanPanelButton_SetButtonIcon(NAME_PREFIX..name, value);
+	end
+
+	if attr == "iconR" or attr == "iconB" or attr == "iconG" then
+		TitanPanelButton_SetButtonIcon(NAME_PREFIX..name, nil,
+			dataobj.iconR, dataobj.iconG, dataobj.iconB);
+	end
+end
+
+--[[ Titan
+NAME: TitanLDBRefreshButton
+DESC: Refresh all text & icon for LDB addons that were successfully registered
+VAR:  None
+OUT:  None
+NOTE:
+- Ensure all the LDB buttons are updated.
+- This is called once x seconds after PEW. This helps close the gap where LDB addons set their text on their PEW event
+:NOTE
+--]]
+function TitanLDBRefreshButton()
+--	TitanDebug("LDB: RefreshButton")
+	for name, obj in ldb:DataObjectIterator() do
+		if obj then
+			LDBToTitan:TitanLDBTextUpdate(_, name, "text", (obj.text or ""), obj)
+			LDBToTitan:TitanLDBIconUpdate(_, name, "icon", (obj.icon or iconTitanDefault), obj)
+		else
+--	TitanDebug("LDB: '"..name.."' no refresh")
+		end
+	end
+end
+
+--[[ Titan
+NAME: LDBToTitan:TitanLDBCreateObject
+DESC: New DO (Data Object) gets created here
+VAR: _ - not used
+VAR: name -  id of the plugin
+VAR: obj - LDB object
+OUT:  None
+NOTE:
+- This is the heart of the LDB to Titan. It reads the LDB DO (Data Object)and creates a Titan plugin.
+- This takes a stricter interpretation of the LDB 1.1 spec rather than guessing what LDB addon developers intended.
+:NOTE
+--]]
+function LDBToTitan:TitanLDBCreateObject(_, name, obj)
+	--TitanDebug("Attempting to register "..name..".");
+
+	-- couple sanity checks
+	if not obj or not name then
+--		TitanDebug(.."LDB request to create Titan plugin was unrecognizable!!!!")
+		return
+	end
+
+	-- anything to pass to the developer / user
+	local notes = ""
+
+	-- sanity check for supported types
+	obj.type = obj.type or "Unknown"
+	local supported = false -- assume failure
+	for idx in ipairs(SupportedDOTypes) do
+		if obj.type and obj.type == SupportedDOTypes[idx] then
+			supported = true
+		end
+	end
+	if supported then
+		-- all is good - continue plugin creation
+	else
+		-- Create enough of a plugin to tell the user / developer
+		-- that this plugin failed miserably
+		local plugin =
+			{
+			self = nil,
+			button = nil,
+			isChild = nil,
+			name = (name or "?"),
+			issue = "Unsupported LDB type '"..obj.type.."'",
+			notes = "",
+			status = TITAN_REGISTER_FAILED,
+			category = "",
+			plugin_type = (obj.type or ""),
+			}
+		TitanUtils_PluginFail(plugin)
+		return -- get out, there is nothing more that can be done
+	end
+
+	--
+	-- Handle the display attributes of the DO and register the appropriate callbacks
+	--
+	-- Init the display elements of the plugin
+	local ldb__label = obj.label or ""
+	local ldb__suffix = obj.suffix or ""
+	local ldb__value = obj.value or ""
+	local ldb__text = obj.text or ""
+	local ldb__icon = obj.icon or iconTitanDefault
+
+	-- if .icon exists honor it and assume the addon may change it
+	if obj.icon then
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_icon", "TitanLDBIconUpdate")
+	end
+
+	-- LAUNCHER text display elements
+	if obj.type == LAUNCHER then
+		if obj.label then
+			ldb.RegisterCallback(self,
+				CALLBACK_PREFIX..name.."_label", "TitanLDBTextUpdate")
+		elseif obj.text then
+			-- This is a 'be nice' check. It technically violates the 1.1 spec.
+			-- Blank the .text so the rest of the routines work
+			ldb__label = obj.text
+			obj.text = ""
+			ldb.RegisterCallback(self,
+				CALLBACK_PREFIX..name.."_text", "TitanLDBTextUpdate")
+			notes = notes.."\n"
+				.."This is a LDB '"..LAUNCHER
+				.."' without .label using .text instead!!!!"
+		end
+	end
+	if Titan__InitializedPEW then
+		notes = notes.."\n"
+			.."Will be registered as single LDB plugin after the normal registration."
+	end
+	-- DATA_SOURCE text display elements
+	if obj.type == DATA_SOURCE then
+		-- .text so always allow it
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_text", "TitanLDBTextUpdate")
+		if obj.label then
+			ldb.RegisterCallback(self,
+				CALLBACK_PREFIX..name.."_label", "TitanLDBTextUpdate")
+		end
+		if obj.suffix then
+			ldb.RegisterCallback(self,
+				CALLBACK_PREFIX..name.."_suffix", "TitanLDBTextUpdate")
+		end
+		if obj.value then
+			ldb.RegisterCallback(self,
+				CALLBACK_PREFIX..name.."_value", "TitanLDBTextUpdate")
+		end
+	end
+
+	--
+	-- These are icon extensions listed within the 1.1 spec
+	--
+	-- support for iconCoords, iconR, iconG, iconB attributes
+	-- Due to the callbacks being fired these can easily affect
+	-- performance, BEWARE when using them !
+	--
+	-- capture the icon coords & color for the Titan plugin
+	if obj.iconCoords then
+		self:TitanLDBIconUpdate(nil, name, "iconCoords", obj.iconCoords, obj)
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_iconCoords", "TitanLDBIconUpdate")
+	end
+	if obj.iconR and obj.iconG and obj.iconB then
+		self:TitanLDBIconUpdate(nil, name, "iconR", obj.iconR, obj)
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_iconR", "TitanLDBIconUpdate")
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_iconG", "TitanLDBIconUpdate")
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_iconB", "TitanLDBIconUpdate")
+	end
+
+	--
+	-- Setup the Titan plugin for this LDB addon
+	--
+
+	-- Create the appropriate Titan registry for the DO
+	local registry = {
+		id = NAME_PREFIX..name,
+		ldb = tostring(obj.type),
+		-- per 1.1 spec if .label exists use it else use data object's name
+		menuText = obj.label or name,
+		buttonTextFunction = "TitanLDBShowText",
+		icon = ldb__icon,
+		iconWidth = 16,
+		controlVariables = {
+			ShowIcon = true,
+			ShowLabelText = true,
+			ShowRegularText = false,
+			ShowColoredText = false,
+			DisplayOnRightSide = true
+		},
+		savedVariables = {
+			ShowIcon = true,
+			ShowLabelText = true,
+			ShowRegularText = true,
+			ShowColoredText = false,
+			DisplayOnRightSide = false
+		},
+		LDBVariables = {
+			value = ldb__value,
+			suffix = ldb__suffix,
+			text = ldb__text,
+			label = ldb__label,
+			name = name,
+			type = (obj.type or ""),
+		},
+		notes = notes,
+		iconCoords = (obj.iconCoords or nil),
+		iconR = (obj.iconR or nil),
+		iconB = (obj.iconB or nil),
+		iconG = (obj.iconG or nil),
+	};
+
+	-- Set the plugin category, if it exists, else default to "General"
+	-- Per the 1.1 LDB spec we check for a tocname attrib first,
+	-- if found we use it, if not we assume that the DO "name"
+	-- attribute is the same as the actual
+	-- addon name, which might not always be the case.
+	-- Titan defaults again to "General" if no categoy is found
+	-- via a check in the menu implementation, later on.
+	local addoncategory, addonversion;
+	local tempname = obj.tocname or name;
+	if IsAddOnLoaded(tempname) then
+		addoncategory = GetAddOnMetadata(tempname, "X-Category");
+		registry["category"]= addoncategory and xcategories[addoncategory] or nil
+		addonversion = GetAddOnMetadata(tempname, "Version");
+		registry["version"]= addonversion or nil;
+	end
+
+	-- Depending on the LDB type set the control and saved Variables appropriately
+	if obj.type == LAUNCHER then
+		-- controls
+		-- one interpretation of the LDB spec is launchers
+		-- should always have an icon.
+		registry["controlVariables"].ShowIcon = false;
+		registry["controlVariables"].ShowRegularText = false; -- no text
+		-- defaults
+		registry["savedVariables"].ShowRegularText = false;
+		registry["savedVariables"].DisplayOnRightSide = true; -- start on right side
+	end
+
+	if obj.type == DATA_SOURCE then
+		-- controls
+		registry["controlVariables"].ShowRegularText = true;
+		-- defaults
+		registry["savedVariables"].ShowRegularText = true;
+	end
+
+	--
+	-- Create the frame for this LDB addon
+	--
+
+	-- Create the Titan Frame as a Combo
+	-- Titan _OnLoad will be used to request the plugin be registered by Titan
+	local newTitanFrame = CreateFrame("Button",
+		"TitanPanel"..NAME_PREFIX..name.."Button",
+		UIParent, "TitanPanelComboTemplate")
+	newTitanFrame.registry = registry
+	newTitanFrame:SetFrameStrata("FULLSCREEN");
+	newTitanFrame:SetToplevel(true);
+	newTitanFrame:RegisterForClicks("LeftButtonUp", "RightButtonUp");
+
+	-- Use the routines given by the DO in this precedence
+	-- tooltip > OnEnter > OnTooltipShow >
+	-- or register a callback in case it is created later. Per the 1.1 LDB spec
+	if obj.tooltip then
+		self:TitanLDBHandleScripts("tooltip", name, nil, obj.tooltip, obj)
+	elseif obj.OnEnter then
+		self:TitanLDBHandleScripts("OnEnter", name, nil, obj.OnEnter, obj)
+	elseif obj.OnTooltipShow then
+		self:TitanLDBHandleScripts("OnTooltipShow", name, nil, obj.OnTooltipShow, obj)
+	else
+		self:TitanLDBHandleScripts("OnEnter", name, nil, nil, obj)
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_OnEnter", "TitanLDBHandleScripts")
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_OnTooltipShow", "TitanLDBHandleScripts")
+	end
+
+	-- Use the OnClick given by the DO
+	-- or register a callback in case it is created later.
+	if obj.OnClick then
+		self:TitanLDBHandleScripts("OnClick", name, nil, obj.OnClick)
+	else
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_OnClick", "TitanLDBHandleScripts")
+	end
+
+	--
+	-- OnDoubleClick is UNDOCUMENTED in the 1.1 spec
+	-- but was implemented by the original developer
+	--
+	-- Use the OnDoubleClick given by the DO
+	-- or register a callback in case it is created later.
+	if obj.OnDoubleClick then
+		self:TitanLDBHandleScripts("OnDoubleClick", name, nil, obj.OnDoubleClick)
+	else
+		ldb.RegisterCallback(self,
+			CALLBACK_PREFIX..name.."_OnDoubleClick", "TitanLDBHandleScripts")
+	end
+
+	-- If plugins have already been registered and loaded then get this one loaded
+	-- This works because the .registry is now set
+	if Titan__InitializedPEW then
+		TitanPanel_PlayerEnteringWorld()
+	end
+end
+
+--[[ Titan
+NAME: LDBToTitan:SetScript
+DESC: OnEvent handler for LDBToTitan
+VAR:  event - string
+VAR:  function
+OUT:  None
+NOTE:
+- PLAYER_LOGIN - Read through all the LDB object created so far and create cooresponding Titan plugins.
+- PLAYER_ENTERING_WORLD - Create a one time timer. This helps ensure the latest values are displayed on the plugin. Some LDB addons create their objects then update the addon values. Titan could have missed the updates as it created & registered the Titan plugin.
+:NOTE
+--]]
+LDBToTitan:SetScript("OnEvent", function(self, event, ...)
+	if (event == "PLAYER_LOGIN") then
+		self:UnregisterEvent("PLAYER_LOGIN")
+		-- In case a LDB plugin is created later...
+		ldb.RegisterCallback(self,
+			"LibDataBroker_DataObjectCreated", "TitanLDBCreateObject")
+
+		-- Register the LDB plugins that have been created so far
+		for name, obj in ldb:DataObjectIterator() do
+			self:TitanLDBCreateObject(nil, name, obj)
+			--TitanDebug("Registered "..name..".");
+		end
+	end
+
+	if (event == "PLAYER_ENTERING_WORLD") then
+		self:UnregisterEvent("PLAYER_ENTERING_WORLD")
+		-- Ensure all plugins (for LDB) are refreshed.
+		-- Some LDB plugins may have updated text, icon, etc
+		-- before the plugin was registered so be nice and schedule a refresh
+		TitanMovable_AdjustTimer("LDBRefresh")
+	end
+	end
+)
\ No newline at end of file
diff --git a/Titan/LibDataBroker-1.1.lua b/Titan/LibDataBroker-1.1.lua
new file mode 100644
index 0000000..4182f2e
--- /dev/null
+++ b/Titan/LibDataBroker-1.1.lua
@@ -0,0 +1,90 @@
+
+assert(LibStub, "LibDataBroker-1.1 requires LibStub")
+assert(LibStub:GetLibrary("CallbackHandler-1.0", true), "LibDataBroker-1.1 requires CallbackHandler-1.0")
+
+local lib, oldminor = LibStub:NewLibrary("LibDataBroker-1.1", 4)
+if not lib then return end
+oldminor = oldminor or 0
+
+
+lib.callbacks = lib.callbacks or LibStub:GetLibrary("CallbackHandler-1.0"):New(lib)
+lib.attributestorage, lib.namestorage, lib.proxystorage = lib.attributestorage or {}, lib.namestorage or {}, lib.proxystorage or {}
+local attributestorage, namestorage, callbacks = lib.attributestorage, lib.namestorage, lib.callbacks
+
+if oldminor < 2 then
+	lib.domt = {
+		__metatable = "access denied",
+		__index = function(self, key) return attributestorage[self] and attributestorage[self][key] end,
+	}
+end
+
+if oldminor < 3 then
+	lib.domt.__newindex = function(self, key, value)
+		if not attributestorage[self] then attributestorage[self] = {} end
+		if attributestorage[self][key] == value then return end
+		attributestorage[self][key] = value
+		local name = namestorage[self]
+		if not name then return end
+		callbacks:Fire("LibDataBroker_AttributeChanged", name, key, value, self)
+		callbacks:Fire("LibDataBroker_AttributeChanged_"..name, name, key, value, self)
+		callbacks:Fire("LibDataBroker_AttributeChanged_"..name.."_"..key, name, key, value, self)
+		callbacks:Fire("LibDataBroker_AttributeChanged__"..key, name, key, value, self)
+	end
+end
+
+if oldminor < 2 then
+	function lib:NewDataObject(name, dataobj)
+		if self.proxystorage[name] then return end
+
+		if dataobj then
+			assert(type(dataobj) == "table", "Invalid dataobj, must be nil or a table")
+			self.attributestorage[dataobj] = {}
+			for i,v in pairs(dataobj) do
+				self.attributestorage[dataobj][i] = v
+				dataobj[i] = nil
+			end
+		end
+		dataobj = setmetatable(dataobj or {}, self.domt)
+		self.proxystorage[name], self.namestorage[dataobj] = dataobj, name
+		self.callbacks:Fire("LibDataBroker_DataObjectCreated", name, dataobj)
+		return dataobj
+	end
+end
+
+if oldminor < 1 then
+	function lib:DataObjectIterator()
+		return pairs(self.proxystorage)
+	end
+
+	function lib:GetDataObjectByName(dataobjectname)
+		return self.proxystorage[dataobjectname]
+	end
+
+	function lib:GetNameByDataObject(dataobject)
+		return self.namestorage[dataobject]
+	end
+end
+
+if oldminor < 4 then
+	local next = pairs(attributestorage)
+	function lib:pairs(dataobject_or_name)
+		local t = type(dataobject_or_name)
+		assert(t == "string" or t == "table", "Usage: ldb:pairs('dataobjectname') or ldb:pairs(dataobject)")
+
+		local dataobj = self.proxystorage[dataobject_or_name] or dataobject_or_name
+		assert(attributestorage[dataobj], "Data object not found")
+
+		return next, attributestorage[dataobj], nil
+	end
+
+	local ipairs_iter = ipairs(attributestorage)
+	function lib:ipairs(dataobject_or_name)
+		local t = type(dataobject_or_name)
+		assert(t == "string" or t == "table", "Usage: ldb:ipairs('dataobjectname') or ldb:ipairs(dataobject)")
+
+		local dataobj = self.proxystorage[dataobject_or_name] or dataobject_or_name
+		assert(attributestorage[dataobj], "Data object not found")
+
+		return ipairs_iter, attributestorage[dataobj], 0
+	end
+end
diff --git a/Titan/TitanAutoHide.lua b/Titan/TitanAutoHide.lua
new file mode 100644
index 0000000..918b81b
--- /dev/null
+++ b/Titan/TitanAutoHide.lua
@@ -0,0 +1,180 @@
+--[[ File
+NAME:TitanAutoHide.lua
+DESC: Contains the routines of AutoHide Titan plugin to auto hide a Titan bar.
+
+Auto hide uses a data driven approach. Rather than seperate routines for each bar, auto hide is implemented in a general manner.
+The tables TitanBarData & AutoHideData hold relevant data needed to control auto hide.
+The index into AutoHideData is the plugin button name given in TitanPanel.xml.
+
+If auto hide is turned on these routines will show / hide the proper bar (and plugins on the bar).
+These routines control the 'push pin' on each bar, if shown.
+:DESC
+--]]
+local AceTimer = LibStub("AceTimer-3.0")
+local L = LibStub("AceLocale-3.0"):GetLocale("Titan", true)
+local Dewdrop = nil
+if AceLibrary and AceLibrary:HasInstance("Dewdrop-2.0") then Dewdrop = AceLibrary("Dewdrop-2.0") end
+
+-- local routines
+--[[ local
+NAME: Titan_AutoHide_SetIcon
+DESC: Set the icon for the plugin.
+VAR: self - The bar
+OUT: None
+--]]
+local function Titan_AutoHide_SetIcon(self)
+	local frame = self:GetName()
+	local bar = AutoHideData[frame].name
+
+	-- Get the icon of the icon template
+	local icon = _G[frame.."Icon"]
+	if (TitanPanelGetVar(bar.."_Hide")) then
+		icon:SetTexture("Interface\\AddOns\\Titan\\Artwork\\TitanPanelPushpinOut")
+	else
+		icon:SetTexture("Interface\\AddOns\\Titan\\Artwork\\TitanPanelPushpinIn")
+	end
+end
+
+-- Event handlers
+--[[ Titan
+NAME: Titan_AutoHide_OnLoad
+DESC: Setup the plugin on the given bar.
+VAR: self - The bar
+OUT: None
+--]]
+function Titan_AutoHide_OnLoad(self)
+	local frame = self:GetName()
+	local bar = AutoHideData[frame].name
+
+	self.registry = {
+		id = "AutoHide_"..bar,
+		category = "Built-ins",
+		version = TITAN_VERSION,
+		menuText = "AutoHide_"..bar,
+		tooltipTitle = L["TITAN_AUTOHIDE_TOOLTIP"],
+		savedVariables = {
+			DisplayOnRightSide = 1,
+			ForceBar = bar,
+		}
+	};
+end
+
+--[[ Titan
+NAME: Titan_AutoHide_OnShow
+DESC: Show the plugin on the given bar.
+VAR: self - The bar
+OUT: None
+--]]
+function Titan_AutoHide_OnShow(self)
+	Titan_AutoHide_SetIcon(self)
+end
+
+--[[ Titan
+NAME: Titan_AutoHide_OnClick
+DESC: Handle button clicks on the given bar.
+VAR: self - The bar
+VAR: button - The mouse button clicked
+OUT:  None
+--]]
+function Titan_AutoHide_OnClick(self, button)
+	if (button == "LeftButton") then
+		Titan_AutoHide_ToggleAutoHide(self);
+	end
+end
+
+-- Auto hide routines
+--[[ Titan
+NAME: Titan_AutoHide_Timers
+DESC: This routine accepts the display bar frame and whether the cursor is entering or leaving. On enter kill the timers that are looking to hide the bar. On leave start the timer to hide the bar.
+VAR: frame - The bar
+VAR: action - "Enter" | "Leave"
+OUT:  None
+--]]
+function Titan_AutoHide_Timers(frame, action)
+	if not frame or not action then
+		return
+	end
+	local bar = (TitanBarData[frame] and TitanBarData[frame].name or nil)
+	local hide = (bar and TitanPanelGetVar(bar.."_Hide") or nil)
+
+	if bar and hide then
+		if (action == "Enter") then
+				AceTimer.CancelAllTimers(frame)
+		end
+		if (action == "Leave") then
+			-- pass the bar as an arg so we can get it back
+			AceTimer.ScheduleRepeatingTimer(frame, Handle_OnUpdateAutoHide, 0.5, frame)
+		end
+	end
+end
+
+--[[ Titan
+NAME: Titan_AutoHide_Init
+DESC: Show / hide the given bar per the user requested settings
+VAR: self - The bar
+OUT: None
+--]]
+function Titan_AutoHide_Init(self)
+	if not self then return end -- sanity check
+
+	local frame = self:GetName()
+	if frame then -- sanity check
+		local bar = AutoHideData[frame].name
+
+		-- Make sure the bar should be processed
+		if (TitanPanelGetVar(bar.."_Show")) then
+			-- Hide / show the bar the plugin is on
+			if (TitanPanelGetVar(bar.."_Hide")) then
+				TitanPanelBarButton_Hide(TITAN_PANEL_DISPLAY_PREFIX..bar);
+			else
+				TitanPanelBarButton_Show(TITAN_PANEL_DISPLAY_PREFIX..bar);
+			end
+		else
+			TitanPanelBarButton_Hide(TITAN_PANEL_DISPLAY_PREFIX..bar);
+		end
+		Titan_AutoHide_SetIcon(self)
+	end
+end
+
+--[[ Titan
+NAME: Titan_AutoHide_ToggleAutoHide
+DESC: Toggle the user requested show / hide setting then show / hide given bar
+VAR: self - The bar
+OUT: None
+--]]
+function Titan_AutoHide_ToggleAutoHide(self)
+	local frame = self:GetName()
+	local bar = AutoHideData[frame].name
+
+	-- toggle the correct auto hide variable
+	TitanPanelToggleVar(bar.."_Hide")
+	-- Hide / show the requested bar
+	Titan_AutoHide_Init(self)
+end
+
+--[[ Titan
+NAME: Handle_OnUpdateAutoHide
+DESC: Hide the bar if the user has auto hide after the cursor leaves the display bar.
+VAR: frame - The bar
+OUT: None
+--]]
+function Handle_OnUpdateAutoHide(frame)
+	if TitanPanelRightClickMenu_IsVisible()
+	or (Tablet20Frame and Tablet20Frame:IsVisible())
+	or (Dewdrop and Dewdrop:IsOpen())then
+		return
+	end
+
+	local data = TitanBarData[frame] or nil
+	if not data then -- sanity check
+		return
+	end
+	local bar = (data.name or nil)
+
+	local hide = TitanPanelGetVar(bar.."_Hide")
+	--
+	if hide then
+		AceTimer.CancelAllTimers(frame)
+		TitanPanelBarButton_Hide(frame)
+	end
+end
diff --git a/Titan/TitanAutoHide.xml b/Titan/TitanAutoHide.xml
new file mode 100644
index 0000000..9627f42
--- /dev/null
+++ b/Titan/TitanAutoHide.xml
@@ -0,0 +1,38 @@
+<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="TitanAutoHide.lua"/>
+	<Frame parent="UIParent">
+		<Frames>
+			<Button name="TitanPanelAutoHideButton" inherits="TitanPanelIconTemplate" frameStrata="FULLSCREEN" toplevel="true">
+				<Scripts>
+					<OnLoad>
+						TitanPanelAutoHideButton_OnLoad(self);
+						TitanPanelButton_OnLoad(self);
+					</OnLoad>
+					<OnShow>
+						TitanPanelAutoHideButton_OnShow();
+					</OnShow>
+					<OnClick>
+						TitanPanelAutoHideButton_OnClick(self, button);
+						TitanPanelButton_OnClick(self, button);
+					</OnClick>
+				</Scripts>
+			</Button>
+			<Button name="TitanPanelAuxAutoHideButton" inherits="TitanPanelIconTemplate" frameStrata="FULLSCREEN" toplevel="true">
+				<Scripts>
+					<OnLoad>
+						TitanPanelAuxAutoHideButton_OnLoad(self);
+						TitanPanelButton_OnLoad(self);
+					</OnLoad>
+					<OnShow>
+						TitanPanelAuxAutoHideButton_OnShow();
+					</OnShow>
+					<OnClick>
+						TitanPanelAuxAutoHideButton_OnClick(self, button);
+						TitanPanelButton_OnClick(self, button);
+					</OnClick>
+				</Scripts>
+			</Button>
+		</Frames>
+	</Frame>
+</Ui>
\ No newline at end of file
diff --git a/Titan/TitanConfig.lua b/Titan/TitanConfig.lua
new file mode 100644
index 0000000..f219c2c
--- /dev/null
+++ b/Titan/TitanConfig.lua
@@ -0,0 +1,1748 @@
+--[[ File
+NAME: TitanConfig.lua
+DESC: This file contains routines used by Titan to show and process the Titan options.
+Titan uses Ace libraries to place the Titan options within the Blizzard option screens.
+
+Most routines in this file are local because they create the Titan options.
+These routines are called first when Titan processes the 'player entering world' event.
+If an options list (skins, extra, etc) is changed by the user then the Ace table needs to be updated and Blizz informed to 'redraw'.
+:DESC
+--]]
+
+local L = LibStub("AceLocale-3.0"):GetLocale(TITAN_ID, true)
+local AceConfigDialog = LibStub("AceConfigDialog-3.0")
+local AceConfigRegistry = LibStub("AceConfigRegistry-3.0")
+local AceConfig = LibStub("AceConfig-3.0")
+
+local TitanSkinToRemove = "None";
+local TitanSkinName, TitanSkinPath = "", "";
+local TitanGlobalProfile = ""
+
+-- Titan local helper funcs
+local function TitanPanel_GetTitle()
+	return GetAddOnMetadata(TITAN_ID, "Title") or L["TITAN_NA"];
+end
+
+local function TitanPanel_GetAuthor()
+	return GetAddOnMetadata(TITAN_ID, "Author") or L["TITAN_NA"];
+end
+
+local function TitanPanel_GetCredits()
+	return GetAddOnMetadata(TITAN_ID, "X-Credits") or L["TITAN_NA"];
+end
+
+local function TitanPanel_GetCategory()
+	return GetAddOnMetadata(TITAN_ID, "X-Category") or L["TITAN_NA"];
+end
+
+local function TitanPanel_GetEmail()
+	return GetAddOnMetadata(TITAN_ID, "X-Email") or L["TITAN_NA"];
+end
+
+local function TitanPanel_GetWebsite()
+	return GetAddOnMetadata(TITAN_ID, "X-Website") or L["TITAN_NA"];
+end
+
+local function TitanPanel_GetLicense()
+	return GetAddOnMetadata(TITAN_ID, "X-License") or L["TITAN_NA"];
+end
+
+--[[ local
+NAME: TitanAdjustPanelScale
+DESC: Set the Tian bars and plugins to the selected scale then adjust other frames as needed.
+VAR: scale - the scale the user has selected for Titan
+OUT: None
+--]]
+local function TitanAdjustPanelScale(scale)
+	Titan_AdjustScale()
+
+	-- Adjust frame positions
+	TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_BOTH, true)
+end
+
+-- About config section
+--[[ local
+NAME: TitanPanel_TicketReload
+DESC: When the user changes the option to adjust for the Blizz ticket frame the UI must be reloaded. Ask the user if they want to do it now.
+VAR:  None
+OUT:  None
+--]]
+local function TitanPanel_TicketReload()
+	StaticPopupDialogs["TITAN_RELOAD"] = {
+		text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"]).."\n\n"
+			..L["TITAN_PANEL_RELOAD"],
+		button1 = ACCEPT,
+		button2 = CANCEL,
+		OnAccept = function(self)
+			TitanPanelToggleVar("TicketAdjust");
+			ReloadUI();
+			end,
+		showAlert = 1,
+		timeout = 0,
+		whileDead = 1,
+		hideOnEscape = 1
+	};
+	StaticPopup_Show("TITAN_RELOAD");
+end
+
+-------------
+-- skins config section
+--[[ local
+NAME: optionsControl
+DESC: Local table to hold the 'about' Titan info in the options.
+--]]
+local optionsControl = {
+	name = L["TITAN_PANEL"],
+	type = "group",
+	args = {
+		confgendesc = {
+			order = 1,
+			type = "description",
+			name = L["TITAN_PANEL_CONFIG_MAIN_LABEL"].."\n\n",
+			cmdHidden = true
+		},
+		confinfodesc = {
+			name = "About",
+			type = "group", inline = true,
+			args = {
+				confversiondesc = {
+				order = 1,
+				type = "description",
+				name = "|cffffd700"..L["TITAN_ABOUT_VERSION"]..": "
+					.._G["GREEN_FONT_COLOR_CODE"]..TitanPanel_GetVersion(),
+				cmdHidden = true
+				},
+				confauthordesc = {
+					order = 2,
+					type = "description",
+					name = "|cffffd700"..L["TITAN_ABOUT_AUTHOR"]..": "
+						.."|cffff8c00"..TitanPanel_GetAuthor(),
+					cmdHidden = true
+				},
+				confcreditsdesc = {
+					order = 3,
+					type = "description",
+					name = "|cffffd700"..L["TITAN_ABOUT_CREDITS"]..": "
+						.._G["HIGHLIGHT_FONT_COLOR_CODE"]..TitanPanel_GetCredits(),
+					cmdHidden = true
+				},
+				confcatdesc = {
+					order = 4,
+					type = "description",
+					name = "|cffffd700"..L["TITAN_ABOUT_CATEGORY"]..": "
+						.._G["HIGHLIGHT_FONT_COLOR_CODE"]..TitanPanel_GetCategory(),
+					cmdHidden = true
+				},
+				confemaildesc = {
+					order = 5,
+					type = "description",
+					name = "|cffffd700"..L["TITAN_ABOUT_EMAIL"]..": "
+						.._G["HIGHLIGHT_FONT_COLOR_CODE"]..TitanPanel_GetEmail(),
+					cmdHidden = true
+				},
+				confwebsitedesc = {
+					order = 6,
+					type = "description",
+					name = "|cffffd700"..L["TITAN_ABOUT_WEB"]..": "
+						.._G["HIGHLIGHT_FONT_COLOR_CODE"]..TitanPanel_GetWebsite(),
+					cmdHidden = true
+				},
+				conflicensedesc = {
+					order = 7,
+					type = "description",
+					name = "|cffffd700"..L["TITAN_ABOUT_LICENSE"]..": "
+						.._G["HIGHLIGHT_FONT_COLOR_CODE"]..TitanPanel_GetLicense(),
+					cmdHidden = true
+				},
+			}
+		}
+	}
+}
+-------------
+
+-------------
+-- transparency config section
+--[[ local
+NAME: optionsTrans
+DESC: Local table to hold the Titan transparency options. Each bar and the tooltip can be set to a different transparancy setting.
+This is the starting shell
+- the option description
+- add the tooltip transparency (at the bottom of the options screen)
+- the code for each bar will be added via another routine.
+:DESC
+--]]
+local optionsTrans = {
+	name = L["TITAN_TRANS_MENU_TEXT"],
+	type = "group",
+	args = {
+		confdesc = {
+				order = 1,
+				type = "description",
+				name = L["TITAN_TRANS_MENU_DESC"].."\n",
+				cmdHidden = true
+			},
+		tooltiptrans = {
+			name = L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"],
+			desc = L["TITAN_TRANS_TOOLTIP_DESC"],
+			order = 50, type = "range", width = "full",
+			min = 0, max = 1, step = 0.01,
+			get = function() return TitanPanelGetVar("TooltipTrans") end,
+			set = function(_, a)
+				TitanPanelSetVar("TooltipTrans", a);
+			end,
+		},
+	},
+ }
+--[[ local
+NAME: TitanPanel_TransOptions
+DESC: This will add each Titan bar to the transparency option table so it can be adjusted by the user. Each bar is shown whether ot not the user has them displayed.
+VAR: args - the table holding the Titan option data
+OUT: None
+--]]
+local function TitanPanel_TransOptions(args)
+	local bar = ""
+	local bar_name = ""
+	local var = ""
+	local vert = nil
+	local position
+	for idx,v in pairs (TitanBarData) do
+		bar = TitanBarData[idx].name
+		vert = TitanBarData[idx].vert
+		position = TitanBarData[idx].order + 10
+		var = bar.."_Transparency"
+		bar_name = TITAN_PANEL_DISPLAY_PREFIX..bar
+		args[bar_name] = {
+			name = TitanBarData[idx].locale_name,
+			desc = "",
+			order = position, type = "range", width = "full",
+			min = 0, max = 1, step = 0.01,
+			get = function(info)
+				local bar = TitanBarData[info[1]].name
+				return TitanPanelGetVar(bar.."_Transparency")
+				end,
+			set = function(info, a)
+				local bar = TitanBarData[info[1]].name
+				_G[info[1]]:SetAlpha(a)
+				TitanPanelSetVar(bar.."_Transparency", a);
+			end,
+		}
+	position = position + 1
+	end
+end
+-------------
+
+-------------
+-- skins config section
+--[[ local
+NAME: TitanPanel_SetCustomTexture
+DESC: Update the skin when the user changes it.
+VAR: path - the file path to the skin
+OUT: None
+--]]
+local function TitanPanel_SetCustomTexture(path)
+	if path ~= TitanPanelGetVar("TexturePath") then
+		TitanPanelSetVar("TexturePath", path);
+		for idx,v in pairs (TitanBarData) do
+			TitanPanel_SetTexture(
+				TITAN_PANEL_DISPLAY_PREFIX..TitanBarData[idx].name,
+				TITAN_PANEL_PLACE_TOP);
+		end
+	end
+end
+
+--[[ local
+NAME: TitanPanel_AddNewSkin
+DESC: Add each skin to the options list. If the user had added custom skins these will be shown as well.
+VAR: skinname - the file name to use
+VAR: skinpath - the file path to use
+OUT:  None
+NOTE:
+- Blizz *does not allow* LUA to access the user file system dynamically so the skins have to be input by hand. Titan can not search for available skins in the Artwork folder.
+- On the flip side a user can add a custom skin to the Titan saved variables then later delete the skin from the file system. This will not cause an error when the user tries to use (show) that skin but Titan will show a 'blank' skin.
+:NOTE
+--]]
+local function TitanPanel_AddNewSkin(skinname, skinpath)
+	-- name and path must be provided
+	if not skinname or not skinpath then return end
+
+	-- name cannot be empty or "None", path cannot be empty
+	if skinname == "" or skinname == L["TITAN_NONE"] or skinpath == "" then
+		return
+	end
+
+	-- Assume the skin is already in the Titan saved variables list
+	local found
+	for _,i in pairs(TitanSkins) do
+		if i.name == skinname or i.path == skinpath then
+			found = true
+			break
+		end
+	end
+
+	-- The skin is new so add it to the Titan saved variables list
+	if not found then
+		table.insert(TitanSkins, {name = skinname, path = skinpath })
+	end
+end
+--[[ local
+NAME: optionsSkins
+DESC: Local table to hold the Titan skins options. Shows default Titan and any custom skins the user has added.
+--]]
+local optionsSkins = {
+	name = L["TITAN_SKINS_TITLE"],
+	type = "group",
+	args = {
+		setskinhdear = {
+			order = 2,
+			type = "header",
+			name = L["TITAN_SKINS_SET_HEADER"],
+		},
+		setskinlist = {
+			order = 3, type = "select",
+			name = L["TITAN_SKINS_LIST_TITLE"],
+			desc = L["TITAN_SKINS_SET_DESC"],
+			get = function() return TitanPanelGetVar("TexturePath") end,
+			set = function(_,v)
+				TitanPanel_SetCustomTexture(v)
+				if TitanSkinToRemove == TitanPanelGetVar("TexturePath") then
+					TitanSkinToRemove = "None"
+				end
+			end,
+			values = function()
+				local Skinlist = {}
+				local v;
+				for _,v in pairs (TitanSkins) do
+					if v.path ~= TitanPanelGetVar("TexturePath") then
+						Skinlist[v.path] = "|cff19ff19"..v.name.."|r"
+					else
+						Skinlist[v.path] = "|cffffff9a"..v.name.."|r"
+					end
+				end
+				table.sort(Skinlist, function(a, b)
+					return string.lower(TitanSkins[a].name)
+						< string.lower(TitanSkins[b].name)
+				end)
+				return Skinlist
+			end,
+		},
+		nulloption1 = {
+			order = 5,
+			type = "description",
+			name = "   ",
+			cmdHidden = true
+		},
+		resetskinhdear = {
+		order = 20,
+		type = "header",
+		name = L["TITAN_SKINS_RESET_HEADER"],
+		},
+		defaultskins = {
+			order = 21,
+			name = L["TITAN_SKINS_RESET_DEFAULTS_TITLE"], type = "execute",
+			desc = L["TITAN_SKINS_RESET_DEFAULTS_DESC"],
+			func = function()
+				TitanSkins = TitanSkinsDefault;
+			end,
+		},
+	}
+}
+--[[ local
+NAME: optionsSkinsCustom
+DESC: Local table to hold the Titan custom skins options that allow a user to add or delete skins.
+- You may not remove the currently used skin
+- or the default one
+- or a Titan default skin (it would only come back...)
+:DESC
+--]]
+local optionsSkinsCustom = {
+	name = L["TITAN_SKINS_TITLE_CUSTOM"],
+	type = "group",
+	args = {
+		confdesc = {
+			order = 1,
+			type = "description",
+			name = L["TITAN_SKINS_MAIN_DESC"].."\n",
+			cmdHidden = true
+		},
+		nulloption1 = {
+			order = 5,
+			type = "description",
+			name = "   ",
+			cmdHidden = true
+		},
+		addskinheader = {
+			order = 10,
+			type = "header",
+			name = L["TITAN_SKINS_NEW_HEADER"],
+		},
+		newskinname = {
+			order = 11,
+			name = L["TITAN_SKINS_NAME_TITLE"],
+			desc = L["TITAN_SKINS_NAME_DESC"],
+			type = "input", width = "full",
+			get = function() return TitanSkinName end,
+			set = function(_,v) TitanSkinName = v end,
+		},
+		newskinpath = {
+			order = 12,
+			name = L["TITAN_SKINS_PATH_TITLE"],
+			desc = L["TITAN_SKINS_PATH_DESC"],
+			type = "input", width = "full",
+			get = function() return TitanSkinPath end,
+			set = function(_,v) TitanSkinPath = TitanSkinsCustomPath..v..TitanSkinsPathEnd end,
+
+		},
+		addnewskin = {
+			order = 13,
+			name = L["TITAN_SKINS_ADD_HEADER"], type = "execute",
+			desc = L["TITAN_SKINS_ADD_DESC"],
+			func = function()
+				if TitanSkinName ~= "" and TitanSkinPath ~= "" then
+					TitanPanel_AddNewSkin(TitanSkinName, TitanSkinPath)
+					TitanSkinName = ""
+					TitanSkinPath = ""
+					-- Config Tables changed!
+					AceConfigRegistry:NotifyChange("Titan Panel Skin Custom")
+				end
+			end,
+		},
+		nulloption2 = {
+			order = 14,
+			type = "description",
+			name = "   ",
+			cmdHidden = true
+		},
+		removeskinheader = {
+			order = 20,
+			type = "header",
+			name = L["TITAN_SKINS_REMOVE_HEADER"],
+		},
+		removeskinlist = {
+			order = 21, type = "select", width = "full",
+			name = L["TITAN_SKINS_REMOVE_HEADER"],
+			desc = L["TITAN_SKINS_REMOVE_DESC"],
+			get = function() return TitanSkinToRemove end,
+			set = function(_,v)
+				TitanSkinToRemove = v
+			end,
+			values = function()
+			local Skinlist = {}
+			local v;
+				for _,v in pairs (TitanSkins) do
+					if v.path ~= TitanPanelGetVar("TexturePath")
+					and v.path ~= "Interface\\AddOns\\Titan\\Artwork\\"
+					and v.titan ~= true
+					then
+						Skinlist[v.path] = "|cff19ff19"..v.name.."|r"
+					end
+					if v.path == TitanSkinToRemove then
+						Skinlist[v.path] = "|cffffff9a"..v.name.."|r"
+					end
+				end
+				if TitanSkinToRemove ~= "None" then
+					Skinlist["None"] = "|cff19ff19"..L["TITAN_NONE"].."|r"
+				else
+					Skinlist["None"] = "|cffffff9a"..L["TITAN_NONE"].."|r"
+				end
+				table.sort(Skinlist, function(a, b)
+					return string.lower(TitanSkins[a].name)
+						< string.lower(TitanSkins[b].name)
+				end)
+					return Skinlist
+			end,
+		},
+		removeskin = {
+			order = 22, type = "execute",
+			name = L["TITAN_SKINS_REMOVE_BUTTON"],
+			desc = L["TITAN_SKINS_REMOVE_BUTTON_DESC"],
+			func = function()
+			if TitanSkinToRemove == "None" then return end
+			local k, v;
+				for k, v in pairs (TitanSkins) do
+					if v.path == TitanSkinToRemove then
+						table.remove(TitanSkins, k)
+						TitanSkinToRemove = "None"
+						-- Config Tables changed!
+						AceConfigRegistry:NotifyChange("Titan Panel Skin Custom")
+						break
+					end
+				end
+			end,
+		},
+		notes_delete = {
+			order = 23,
+			type = "description",
+			name = L["TITAN_SKINS_REMOVE_NOTES"].."\n",
+			cmdHidden = true
+		},
+		nulloption4 = {
+			order = 24,
+			type = "description",
+			name = "   ",
+			cmdHidden = true
+		},
+	}
+}
+-------------
+
+-------------
+-- UI scale config section
+--[[ local
+NAME: optionsUIScale
+DESC: Local table to hold the Titan options that allow a user to adjust:
+- UI scale
+- Titan scale for bars
+- Spacing between Titan plugins (right side)
+- Spacing between Titan icons (left side)
+- Titan tooltip font scale (bar and plugins)
+- Toggle the tooltip font scale (allow Titan or Blizz to control)
+- Set Titan font (bar and plugins)
+- Set Titan font size
+- Set Titan bar strata (tells Blizz which frames could go over Titan bar (and plugins)
+:DESC
+--]]
+local optionsUIScale = {
+	name = L["TITAN_UISCALE_MENU_TEXT"],
+	type = "group",
+	args = {
+		confdesc = {
+			order = 1,
+			type = "description",
+			name = L["TITAN_UISCALE_MENU_DESC"].."\n",
+			cmdHidden = true
+		},
+		uiscale = {
+			name = L["TITAN_UISCALE_CONTROL_TITLE_UI"],
+			desc = L["TITAN_UISCALE_SLIDER_DESC"],
+			order = 2, type = "range", width = "full",
+			min = 0.64, max = 1, step = 0.01,
+			get = function() return UIParent:GetScale() end,
+			set = function(_, a)
+				SetCVar("useUiScale", 1);
+				SetCVar("uiScale", a, "uiScale");
+			end,
+		},
+		panelscale = {
+			name = L["TITAN_UISCALE_CONTROL_TITLE_PANEL"],
+			desc = L["TITAN_UISCALE_PANEL_SLIDER_DESC"],
+			order = 3, type = "range", width = "full",
+			min = 0.75, max = 1.25, step = 0.01,
+			get = function() return TitanPanelGetVar("Scale") end,
+			set = function(_, a)
+				if not InCombatLockdown() then
+					TitanPanelSetVar("Scale", a);
+					TitanAdjustPanelScale(a)
+				end
+			end,
+			disabled = function()
+				if InCombatLockdown() then
+					return true
+				end
+				return false
+			end,
+		},
+		buttonspacing = {
+			name = L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"],
+			desc = L["TITAN_UISCALE_BUTTON_SLIDER_DESC"],
+			order = 4, type = "range", width = "full",
+			min = 5, max = 80, step = 1,
+			get = function() return TitanPanelGetVar("ButtonSpacing") end,
+			set = function(_, a)
+				TitanPanelSetVar("ButtonSpacing", a);
+				TitanPanel_InitPanelButtons();
+			end,
+		},
+		iconspacing = { -- right side plugins
+			name = L["TITAN_UISCALE_CONTROL_TITLE_ICON"],
+			desc = L["TITAN_UISCALE_ICON_SLIDER_DESC"],
+			order = 5, type = "range", width = "full",
+			min = 0, max = 20, step = 1,
+			get = function() return TitanPanelGetVar("IconSpacing") end,
+			set = function(_, a)
+				TitanPanelSetVar("IconSpacing", a);
+				TitanPanel_InitPanelButtons();
+			end,
+		},
+		tooltipfont = {
+			name = L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"],
+			desc = L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"],
+			order = 10, type = "range", width = "full",
+			min = 0.5, max = 1.3, step = 0.01,
+			get = function() return TitanPanelGetVar("TooltipFont") end,
+			set = function(_, a)
+				TitanPanelSetVar("TooltipFont", a);
+			end,
+		},
+		tooltipfontdisable = {
+			name = L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"],
+			desc = L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"],
+			order = 11, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("DisableTooltipFont") end,
+			set = function()
+				TitanPanelToggleVar("DisableTooltipFont");
+			end,
+		},
+		fontselection = {
+			name = L["TITAN_PANEL_MENU_LSM_FONTS"],
+			desc = L["TITAN_PANEL_MENU_LSM_FONTS_DESC"],
+			order = 12, type = "select",
+			dialogControl = "LSM30_Font",
+			get = function()
+				return TitanPanelGetVar("FontName")
+			end,
+			set = function(_, v)
+				TitanPanelSetVar("FontName", v)
+				TitanSetPanelFont(v, TitanPanelGetVar("FontSize"))
+			end,
+			values = AceGUIWidgetLSMlists.font,
+		},
+		fontsize = {
+			name = L["TITAN_PANEL_MENU_FONT_SIZE"],
+			desc = L["TITAN_PANEL_MENU_FONT_SIZE_DESC"],
+			order = 13, type = "range",
+			min = 7, max = 15, step = 1,
+			get = function() return TitanPanelGetVar("FontSize") end,
+			set = function(_, v)
+				TitanPanelSetVar("FontSize", v);
+				TitanSetPanelFont(TitanPanelGetVar("FontName"), v)
+			end,
+		},
+		panelstrata = {
+			name = L["TITAN_PANEL_MENU_FRAME_STRATA"],
+			desc = L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"],
+			order = 14, type = "select",
+			get = function()
+				return TitanPanelGetVar("FrameStrata")
+			end,
+			set = function(_, v)
+				TitanPanelSetVar("FrameStrata", v)
+				TitanVariables_SetPanelStrata(v)
+			end,
+			values = {
+			["BACKGROUND"] = "BACKGROUND",
+			["LOW"] = "LOW",
+			["MEDIUM"] = "MEDIUM",
+			["HIGH"] = "HIGH",
+			["DIALOG"] = "DIALOG",
+			["FULLSCREEN"] = "FULLSCREEN",
+			},
+		},
+	}
+}
+-------------
+
+-------------
+-- Bar control config section
+--[[ local
+NAME: optionsBars
+DESC: Bar control for the main (top) bars:
+Each bar:
+- Show
+- Auto hide
+- Center text (plugins)
+Main (top) controls:
+- Disable screen adjust - allows character frame and minimap to be 'over' the Titan bars
+- Disable minimap adjust - allows the minimap to be 'over' the Titan bars
+- Ticket frame adjust - adjusts the Blizz open ticket frame to be under the Titan bar(s)
+:DESC
+--]]
+local optionsBars = {
+	name = L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"],
+	type = "group",
+	args = {
+		confdesc1 = {
+			order = 100,
+			type = "header",
+			name = L["TITAN_PANEL_MENU_TOP"],
+		},
+		optiontop = {
+			name = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			desc = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			order = 101, type = "toggle", width = "full",
+			get = function() return (TitanPanelGetVar("Bar_Show")) end,
+			set = function()
+					TitanPanelToggleVar("Bar_Show")
+					TitanPanelBarButton_DisplayBarsWanted()
+					end,
+		},
+		optiontophide = {
+			name = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			desc = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			order = 103, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("Bar_Hide") end,
+			set = function()
+				TitanPanelBarButton_ToggleAutoHide(TITAN_PANEL_DISPLAY_PREFIX.."Bar")
+			end,
+		},
+		optiontopcenter = {
+			name = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			desc = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			order = 104, type = "toggle", width = "full",
+			get = function()
+				return (TitanPanelGetVar("Bar_Align") == TITAN_PANEL_BUTTONS_ALIGN_CENTER)
+			end,
+			set = function()
+			local tmp = TitanPanelGetVar("Bar_Align");
+			TitanPanelBarButton_ToggleAlign("Bar_Align");
+--			TitanDebug("Bar c: "..tmp.." "..TitanPanelGetVar("Bar_Align"));
+			end,
+		},
+		confdesc2 = {
+			order = 200,
+			type = "header",
+			name = L["TITAN_PANEL_MENU_TOP2"],
+		},
+		optionbottom = {
+			name = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			desc = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			order = 201, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("Bar2_Show") end,
+			set = function()
+					TitanPanelToggleVar("Bar2_Show")
+					TitanPanelBarButton_DisplayBarsWanted()
+					end,
+		},
+		optionbottomhide = {
+			name = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			desc = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			order = 203, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("Bar2_Hide") end,
+			set = function()
+				TitanPanelBarButton_ToggleAutoHide(TITAN_PANEL_DISPLAY_PREFIX.."Bar2");
+			end,
+		},
+		optionbottomcenter = {
+			name = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			desc = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			order = 204, type = "toggle", width = "full",
+			get = function()
+				return (TitanPanelGetVar("Bar2_Align") == TITAN_PANEL_BUTTONS_ALIGN_CENTER)
+			end,
+			set = function() TitanPanelBarButton_ToggleAlign("Bar2_Align"); end,
+		},
+		confdesc3 = {
+			order = 300,
+			type = "header",
+			name = L["TITAN_PANEL_OPTIONS"],
+		},
+		optiontopscreen = {
+			name = L["TITAN_PANEL_MENU_DISABLE_PUSH"],
+			desc = L["TITAN_PANEL_MENU_DISABLE_PUSH"],
+			order = 301, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("ScreenAdjust") end,
+			set = function() TitanPanelBarButton_ToggleScreenAdjust(); end,
+		},
+		optionminimap = {
+			name = L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"],
+			desc = L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"],
+			order = 302, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("MinimapAdjust") end,
+			set = function() TitanPanelToggleVar("MinimapAdjust"); end,
+		},
+		optiontickets = {
+			name = L["TITAN_PANEL_MENU_DISABLE_TICKET"].." "
+				.._G["GREEN_FONT_COLOR_CODE"]..L["TITAN_PANEL_MENU_RELOADUI"],
+			desc = L["TITAN_PANEL_MENU_DISABLE_TICKET"].." "
+				.._G["GREEN_FONT_COLOR_CODE"]..L["TITAN_PANEL_MENU_RELOADUI"],
+			order = 305, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("TicketAdjust"); end,
+			set = function() TitanPanel_TicketReload() end,
+		},
+	}
+}
+
+--[[ local
+NAME: optionsAuxBars
+DESC: Bar control for the aux (bottom) bars:
+Each bar:
+- Show
+- Auto hide
+- Center text (plugins)
+Main (top) controls:
+- Disable screen adjust - allows action button bar to be 'over' the Titan bars
+- Log adjust - move the chat frame to be 'above' the Titan bars
+- Bag adjust - move the bag frames to be 'above' the Titan bars
+:DESC
+--]]
+local optionsAuxBars = {
+	name = L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"],
+	type = "group",
+	args = {
+		confdesc1 = {
+			order = 100,
+			type = "header",
+			name = L["TITAN_PANEL_MENU_BOTTOM"],
+		},
+		optiontop = {
+			name = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			desc = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			order = 101, type = "toggle", width = "full",
+			get = function() return (TitanPanelGetVar("AuxBar_Show")) end,
+			set = function()
+					TitanPanelToggleVar("AuxBar_Show")
+					TitanPanelBarButton_DisplayBarsWanted()
+					end,
+		},
+		optiontophide = {
+			name = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			desc = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			order = 103, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("AuxBar_Hide") end,
+			set = function()
+				TitanPanelBarButton_ToggleAutoHide(TITAN_PANEL_DISPLAY_PREFIX.."AuxBar");
+			end,
+		},
+		optiontopcenter = {
+			name = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			desc = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			order = 104, type = "toggle", width = "full",
+			get = function()
+				return (TitanPanelGetVar("AuxBar_Align") == TITAN_PANEL_BUTTONS_ALIGN_CENTER)
+			end,
+			set = function() TitanPanelBarButton_ToggleAlign("AuxBar_Align"); end,
+		},
+		confdesc2 = {
+				order = 200,
+				type = "header",
+				name = L["TITAN_PANEL_MENU_BOTTOM2"],
+		},
+		optionbottom = {
+			name = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			desc = L["TITAN_PANEL_MENU_DISPLAY_BAR"],
+			order = 201, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("AuxBar2_Show") end,
+			set = function()
+					TitanPanelToggleVar("AuxBar2_Show")
+					TitanPanelBarButton_DisplayBarsWanted()
+					end,
+		},
+		optionbottomhide = {
+			name = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			desc = L["TITAN_PANEL_MENU_AUTOHIDE"],
+			order = 203, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("AuxBar2_Hide") end,
+			set = function()
+				TitanPanelBarButton_ToggleAutoHide(TITAN_PANEL_DISPLAY_PREFIX.."AuxBar2");
+			end,
+		},
+		optionbottomcenter = {
+			name = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			desc = L["TITAN_PANEL_MENU_CENTER_TEXT"],
+			order = 204, type = "toggle", width = "full",
+			get = function()
+				return (TitanPanelGetVar("AuxBar2_Align") == TITAN_PANEL_BUTTONS_ALIGN_CENTER)
+			end,
+			set = function() TitanPanelBarButton_ToggleAlign("AuxBar2_Align"); end,
+		},
+		confdesc3 = {
+			order = 300,
+			type = "header",
+			name = L["TITAN_PANEL_OPTIONS"],
+		},
+		optionbottomscreen = {
+			name = L["TITAN_PANEL_MENU_DISABLE_PUSH"],
+			desc = L["TITAN_PANEL_MENU_DISABLE_PUSH"],
+			order = 301, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("AuxScreenAdjust") end,
+			set = function() TitanPanelBarButton_ToggleAuxScreenAdjust(); end,
+		},
+		optionlog = {
+			name = L["TITAN_PANEL_MENU_DISABLE_LOGS"],
+			desc = L["TITAN_PANEL_MENU_DISABLE_LOGS"],
+			order = 303, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("LogAdjust") end,
+			set = function() TitanPanelToggleVar("LogAdjust"); end,
+		},
+		optionbags = {
+			name = L["TITAN_PANEL_MENU_DISABLE_BAGS"],
+			desc = L["TITAN_PANEL_MENU_DISABLE_BAGS"],
+			order = 304, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("BagAdjust") end,
+			set = function() TitanPanelToggleVar("BagAdjust"); end,
+		},
+	}
+}
+-------------
+
+-------------
+-- General Titan config section
+--[[ local
+NAME: optionsFrames
+DESC: Show the general Tian options that hte user can change:
+Tooltips:
+- Hide in combat
+- Show (or not)
+Frames (bars):
+- Lock buttons (plugins) - do not allow plugins to be moved via drag & drop. Shift left / right is still allowwed.
+- Show plugin versions - show the version in the tooltips
+Actions:
+- Force LDB laucnhers to right side - This will move all converted LDB plugins of type launcher to the right side of the Titan bar.
+- Refresh plugins - This can be used when a plugin has not updated its text. It may allow a plugin to show if it is not visible but the user has selected show.
+- Reset Titan to default - used when the user wants to reset Titan options to a fresh install state. No plugins are removed by this.
+:DESC
+--]]
+local optionsFrames = {
+	name = L["TITAN_PANEL_MENU_OPTIONS"],
+	type = "group",
+	args = {
+		confdesc2 = {
+			order = 200,
+			type = "header",
+			name = L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"],
+		},
+		optiontooltip = {
+			name = L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"],
+--			desc = L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"],
+			order = 201, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("ToolTipsShown") end,
+			set = function() TitanPanelToggleVar("ToolTipsShown"); end,
+		},
+		optiontooltipcombat = {
+			name = L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"],
+--			desc = L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"],
+			order = 201, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("HideTipsInCombat") end,
+			set = function() TitanPanelToggleVar("HideTipsInCombat"); end,
+		},
+		confdesc = {
+			order = 300,
+			type = "header",
+			name = L["TITAN_PANEL_MENU_OPTIONS_FRAMES"],
+		},
+		optionlock = {
+			name = L["TITAN_PANEL_MENU_LOCK_BUTTONS"],
+			desc = L["TITAN_PANEL_MENU_LOCK_BUTTONS"],
+			order = 301, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("LockButtons") end,
+			set = function() TitanPanelToggleVar("LockButtons") end,
+		},
+		optionversions = {
+			name = L["TITAN_PANEL_MENU_VERSION_SHOWN"],
+--			desc = L["TITAN_PANEL_MENU_VERSION_SHOWN"],
+			order = 302, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("VersionShown") end,
+			set = function() TitanPanelToggleVar("VersionShown") end,
+		},
+		autohidelock = {
+			name = L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"],
+			desc = "Lock_auto_hide",
+			order = 303, type = "toggle", width = "full",
+			get = function() return TitanPanelGetVar("LockAutoHideInCombat") end,
+			set = function() TitanPanelToggleVar("LockAutoHideInCombat") end,
+		},
+		space_400_1 = {
+			order = 400,
+			type = "description",
+			name = "  ",
+			cmdHidden = true,
+		},
+		optionlaunchers = {
+			name = L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"],
+			order = 401, type = "execute", width = "full",
+			func = function() TitanPanelBarButton_ForceLDBLaunchersRight() end,
+		},
+		space_500_1 = {
+			order = 500,
+			type = "description",
+			name = "  ",
+			cmdHidden = true,
+		},
+		pluginreset = {
+			name = L["TITAN_PANEL_MENU_PLUGIN_RESET"],
+			desc = L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"],
+			order = 501, type = "execute", width = "full",
+			func = function() TitanPanel_InitPanelButtons() end,
+		},
+		space_600_1 = {
+			order = 600,
+			type = "description",
+			name = "  ",
+			cmdHidden = true,
+		},
+		optionreset = {
+			name = L["TITAN_PANEL_MENU_RESET"].." "
+				.._G["GREEN_FONT_COLOR_CODE"]
+				..L["TITAN_PANEL_MENU_RELOADUI"],
+			order = 601, type = "execute", width = "full",
+			func = function() TitanPanel_ResetToDefault() end,
+		}
+	}
+ }
+-------------
+
+-------------
+-- attempted plugins config section
+--[[ local
+NAME: optionsAddonAttempts
+DESC: This is the table shell. The plugin info will be added by another routine.
+--]]
+local optionsAddonAttempts = {
+	name = L["TITAN_PANEL_ATTEMPTS"],
+	type = "group",
+	args = {}
+}
+--[[ local
+NAME: TitanUpdateAddonAttempts
+DESC: Show the each plugin that attempted to register with Titan. This can be used by plugin developers as the create / update plugins (Titan or LDB). It can also be used by user to attempt to figure out why a plugin is not shown or to report an issue to Titan.
+VAR:  None
+OUT:  None
+NOTE:
+- This is called after the plugins are registered in the 'player entering world' event. It can be called again as plugins registered.
+- Any plugins that attempted to register (no child frames though) are shown. See the Titan Utils section for more details on plugin registration.
+- This option page is for display only. The user can take not action.
+:NOTE
+--]]
+local function TitanUpdateAddonAttempts()
+	local args = optionsAddonAttempts.args
+	local plug_in = nil
+
+	wipe(args)
+
+	args["desc"] = {
+		order = 0,
+		type = "description",
+		name = L["TITAN_PANEL_ATTEMPTS_DESC"],
+		cmdHidden = true
+	}
+	for idx, value in pairs(TitanPluginToBeRegistered) do
+		if TitanPluginToBeRegistered[idx]
+		then
+			local num = tostring(idx)
+			local button = TitanPluginToBeRegistered[idx].button
+			local name = (TitanPluginToBeRegistered[idx].name or "?")
+			local reason = TitanPluginToBeRegistered[idx].status
+			local issue = TitanPluginToBeRegistered[idx].issue
+			local notes = TitanPluginToBeRegistered[idx].notes or ""
+			local category = TitanPluginToBeRegistered[idx].category
+			local ptype = TitanPluginToBeRegistered[idx].plugin_type
+			local btype = TitanPanelButton_GetType(idx)
+			local title = TitanPluginToBeRegistered[idx].name
+			local isChild = TitanPluginToBeRegistered[idx].isChild and true or false
+			if reason ~= TITAN_REGISTERED then
+				title = TitanUtils_GetRedText(title)
+				issue = TitanUtils_GetRedText(issue)
+			end
+
+			if isChild then
+				-- Do not show. A child is part of (within) another plugin
+				-- showing it here would be confusing to the 'normal' user.
+				-- A plugin in author would know to look at the
+				-- TitanPluginToBeRegistered array directly.
+			else
+				args[num] = {
+					type = "group",
+					name = title,
+					order = idx,
+					args = {
+						name ={
+							type = "description",
+							name = TitanUtils_GetGoldText("")..name,
+							cmdHidden = true,
+							order = 1,
+						},
+						reason = {
+							type = "description",
+							name = TitanUtils_GetGoldText(L["TITAN_PANEL_ATTEMPTS_STATUS"]..": ")..reason,
+							cmdHidden = true,
+							order = 2,
+						},
+						issue = {
+							type = "description",
+							name = TitanUtils_GetGoldText(L["TITAN_PANEL_ATTEMPTS_ISSUE"]..": \n")..issue,
+							cmdHidden = true,
+							order = 3,
+						},
+						notes = {
+							type = "description",
+							name = TitanUtils_GetGoldText(L["TITAN_PANEL_ATTEMPTS_NOTES"]..": \n")..notes,
+							cmdHidden = true,
+							order = 4,
+						},
+						sp_1 = {
+							type = "description",
+							name = "",
+							cmdHidden = true,
+							order = 5,
+						},
+						category = {
+							type = "description",
+							name = TitanUtils_GetGoldText(L["TITAN_PANEL_ATTEMPTS_CATEGORY"]..": ")..category,
+							cmdHidden = true,
+							order = 10,
+						},
+						ptype = {
+							type = "description",
+							name = TitanUtils_GetGoldText(L["TITAN_PANEL_ATTEMPTS_TYPE"]..": ")..ptype, --.." "..btype,
+							cmdHidden = true,
+							order = 11,
+						},
+						button = {
+							type = "description",
+							name = TitanUtils_GetGoldText(L["TITAN_PANEL_ATTEMPTS_BUTTON"]..": ")..button,
+							cmdHidden = true,
+							order = 12,
+						},
+						num_val = {
+							type = "description",
+							name = TitanUtils_GetGoldText(L["TITAN_PANEL_ATTEMPTS_TABLE"]..": ")..num,
+							cmdHidden = true,
+							order = 13,
+						},
+					}
+				}
+			end
+		end
+	end
+
+	-- Config Tables changed!
+	AceConfigRegistry:NotifyChange(L["TITAN_PANEL"])
+end
+-------------
+
+-------------
+-- extra plugins config section
+--[[ local
+NAME: optionsExtras
+DESC: This is the table shell. The plugin info will be added by another routine.
+--]]
+local optionsExtras = {
+	name = L["TITAN_PANEL_EXTRAS"],
+	type = "group",
+	args = {}
+ }
+--[[ local
+NAME: TitanUpdateAddonAttempts
+DESC: Show plugins that are not registered (loaded) but have config data. The data can be deleted by the user.
+VAR:  None
+OUT:  None
+NOTE:
+- As users change the plugins they use the old ones still have saved variable data stored by Titan.
+- The old plugin data can be removed by the user when they will not longer use that plugin.
+- This routine is called to 'redraw' the list as a user deletes data.
+- A message is sent to chat that the plugin data has been deleted.
+:NOTE
+--]]
+local function TitanUpdateExtras()
+	local args = optionsExtras.args
+	local plug_in = nil
+
+	wipe(args)
+
+	args["desc"] = {
+		order = 1,
+		type = "description",
+		name = L["TITAN_PANEL_EXTRAS_DESC"].."\n",
+		cmdHidden = true
+	}
+	for idx, value in pairs(TitanPluginExtras) do
+		if TitanPluginExtras[idx] then
+			local num = TitanPluginExtras[idx].num
+			local name = TitanPluginExtras[idx].id
+			args[name] = {
+				type = "group",
+				name = TitanUtils_GetGoldText(tostring(num)..": "..(name or "?")),
+				order = idx,
+				args = {
+					name = {
+						type = "description",
+						name = TitanUtils_GetGoldText(name or "?"),
+						cmdHidden = true,
+						order = 10,
+					},
+					optionreset = {
+						name = L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"],
+						order = 15, type = "execute", width = "full",
+						func = function(info, v)
+							TitanPluginSettings[info[1]] = nil -- delete the config entry
+							TitanPrint(
+								" '"..info[1].."' "..L["TITAN_PANEL_EXTRAS_DELETE_MSG"]
+								, "info")
+							TitanVariables_ExtraPluginSettings() -- rebuild the list
+							TitanUpdateExtras() -- rebuild the options config
+							AceConfigRegistry:NotifyChange("Titan Panel Addon Extras") -- tell Ace to redraw
+						end,
+					},
+				}
+			}
+		end
+	end
+
+	AceConfigRegistry:NotifyChange("Titan Panel Addon Extras")
+end
+-------------
+
+--[[ local
+NAME: optionsChars
+DESC: This is the table shell. The toon info will be added by another routine.
+--]]
+local optionsChars = {
+	name = "Titan "..L["TITAN_PANEL_MENU_PROFILES"],
+	type = "group",
+	args = {}
+}
+--[[ local
+NAME: TitanUpdateChars
+DESC: Allow the user to delete toon data (just not the one they are logged into).
+VAR:  None
+OUT:  None
+NOTE:
+- Users can delete toons but the saved variable data is still stored by Titan.
+- The old toon data can be removed by the user.
+- This routine is called to 'redraw' the list as a user deletes toon data.
+- A message is sent to chat that the plugin data has been deleted.
+:NOTE
+--]]
+local function TitanUpdateChars()
+	local players = {};
+	-- Rip through the players (with server name) to sort them
+	for index, id in pairs(TitanSettings.Players) do
+		table.insert(players, index);
+	end
+
+	-- set up the options for the user
+	local args = optionsChars.args
+	local plug_in = nil
+
+	wipe(args)
+
+	args["desc"] = {
+		order = 1,
+		type = "description",
+		name = L["TITAN_PANEL_CHARS_DESC"].."\n",
+		cmdHidden = true,
+	}
+	args["custom_header"] = {
+		order = 10,
+		type = "header",
+		name = L["TITAN_PANEL_MENU_PROFILE_CUSTOM"].."\n",
+		cmdHidden = true,
+	}
+	args["custom_save"] = {
+		order = 11,
+		type = "execute",
+		name = L["TITAN_PANEL_MENU_SAVE_SETTINGS"].."\n",
+		func = function(info, v)
+			TitanPanel_SaveCustomProfile()
+			TitanUpdateChars() -- rebuild the toons
+		end,
+	}
+	args["sp_1"] = {
+		type = "description",
+		name = "",
+		cmdHidden = true,
+		order = 12,
+	}
+	args["global_header"] = {
+		order = 20,
+		type = "header",
+		name = L["TITAN_PANEL_GLOBAL"],
+		cmdHidden = true,
+	}
+	args["global_use"] = {
+		order = 21, type = "toggle", width = "full",
+		name = L["TITAN_PANEL_GLOBAL_USE"],
+		desc = L["TITAN_PANEL_GLOBAL_USE_DESC"],
+		get = function() return TitanAllGetVar("GlobalProfileUse") end,
+		set = function()
+			TitanUtils_SetGlobalProfile(not TitanAllGetVar("GlobalProfileUse"), toon)
+			TitanUpdateChars() -- rebuild the toons
+			AceConfigRegistry:NotifyChange("Titan Panel Addon Chars")
+		end,
+	}
+	args["global_name"] = {
+		order = 22, type = "description", width = "full",
+		name = L["TITAN_PANEL_GLOBAL_PROFILE"]..": "..TitanUtils_GetGoldText(TitanAllGetVar("GlobalProfileName") or "?"),
+	}
+	args["sp_20"] = {
+		type = "description",
+		name = "",
+		cmdHidden = true,
+		order = 23,
+	}
+	args["profile_header"] = {
+		order = 30,
+		type = "header",
+		name = L["TITAN_PANEL_MENU_PROFILES"].."\n",
+		cmdHidden = true
+	}
+	for idx, value in pairs(players) do
+		local name = (players[idx] or "?")
+		local s, e, ident, server, player
+		local fancy_name = ""
+		local disallow = false
+		disallow = -- looks weird but we need to force a true or Ace complains
+			((name == TitanSettings.Player)
+						or ((name == TitanAllGetVar("GlobalProfileName"))
+							and (TitanAllGetVar("GlobalProfileUse")))
+			) and true or false
+
+		if name then
+			-- color code the name
+			-- - gold for normal profiles
+			-- - green for custom profiles
+			player, server = TitanUtils_ParseName(name)
+			-- handle custom profiles here
+			if server == TITAN_CUSTOM_PROFILE_POSTFIX then
+				fancy_name = TitanUtils_GetGreenText((name or "?"))
+			else
+				fancy_name = TitanUtils_GetGoldText((name or "?"))
+			end
+			-- end color code
+			args[name] = {
+				type = "group",
+				name = fancy_name,
+				desc = "",
+				order = 40,
+				args = {
+					name = {
+						type = "header",
+						name = TitanUtils_GetGoldText(name or "?"),
+						cmdHidden = true,
+						order = 10,
+					},
+					sp_1 = {
+						type = "description",
+						name = "",
+						cmdHidden = true,
+						order = 11,
+					},
+					optionload = {
+						name = L["TITAN_PANEL_MENU_LOAD_SETTINGS"],
+						order = 20, type = "execute", width = "full",
+						func = function(info, v)
+							TitanVariables_UseSettings(info[1], TITAN_PROFILE_USE)
+						end,
+						-- does not make sense to load current character profile or global profile
+						disabled = disallow,
+					},
+					sp_20 = {
+						type = "description",
+						name = "",
+						cmdHidden = true,
+						order = 21,
+					},
+					optionreset = {
+						name = L["TITAN_PANEL_MENU_DELETE_SETTINGS"],
+						order = 30, type = "execute", width = "full",
+						func = function(info, v)
+							TitanSettings.Players[info[1]] = nil -- delete the config entry
+							TitanPrint(
+								L["TITAN_PANEL_MENU_PROFILE"]
+								..info[1]
+								..L["TITAN_PANEL_MENU_PROFILE_DELETED"]
+								, "info")
+							if name == TitanAllGetVar("GlobalProfileName") then
+								TitanAllSetVar("GlobalProfileName", TITAN_PROFILE_NONE)
+							end
+							TitanUpdateChars() -- rebuild the toons
+							AceConfigRegistry:NotifyChange("Titan Panel Addon Chars")
+						end,
+						-- can not delete current character profile or global profile
+						disabled = disallow,
+					},
+					sp_30 = {
+						type = "description",
+						name = "",
+						cmdHidden = true,
+						order = 31,
+					},
+					sp_31 = {
+						type = "description",
+						name = "",
+						cmdHidden = true,
+						order = 32,
+					},
+					global_header = {
+						order = 40,
+						type = "header",
+						name = "Global", --L["TITAN_PANEL_MENU_VERSION_SHOWN"],
+						cmdHidden = true,
+					},
+					use_as_global = {
+						order = 41, type = "toggle", width = "full",
+						name = L["TITAN_PANEL_GLOBAL_USE_AS"],
+						get = function() return TitanPanelGetVar("GlobalProfileName") == name end,
+						set = function()
+							if TitanPanelGetVar("GlobalProfileName") == name then
+								-- Was unchecked so clear the saved var
+								TitanAllSetVar("GlobalProfileName", TITAN_PROFILE_NONE)
+							else
+								-- Was checked so set the saved var
+								TitanAllSetVar("GlobalProfileName", name)
+							end
+							if TitanAllGetVar("GlobalProfileUse") then
+								-- Use whatever toon the user picked, if not use current toon
+								if TitanAllGetVar("GlobalProfileName") == TITAN_PROFILE_NONE then
+									TitanAllSetVar("GlobalProfileName", TitanSettings.Player)
+								end
+								TitanVariables_UseSettings(TitanAllGetVar("GlobalProfileName"), TITAN_PROFILE_USE)
+								TitanPrint(
+									L["TITAN_PANEL_MENU_PROFILE"]
+									..":"..(TitanAllGetVar("GlobalProfileName") or "?")
+									..": "..L["TITAN_PANEL_GLOBAL_RESET_PART"].."..."
+									, "info")
+							else
+								--
+							end
+							TitanUpdateChars()
+							AceConfigRegistry:NotifyChange("Titan Panel Addon Chars")
+						end,
+						-- can not uncheck current global profile
+						disabled = disallow,
+					},
+					sp_40 = {
+						type = "description",
+						name = "",
+						cmdHidden = true,
+						order = 42,
+					},
+				},
+			}
+		end
+	end
+
+	-- tell the options screen there is a new list
+	AceConfigRegistry:NotifyChange("Titan Panel Addon Chars")
+end
+-------------
+
+-------------
+-- plugin control config section
+--[[ local
+NAME: optionsAddons
+DESC: This is the table shell. The plugin controls will be added by another routine.
+--]]
+local optionsAddons = {
+	name = "Titan "..L["TITAN_PANEL_MENU_PLUGINS"],
+	type = "group",
+	args = {}
+}
+--[[ local
+NAME: TitanUpdateConfigAddons
+DESC: Allow the user to control each plugin registered to Titan.
+Controls honored from the plugin .registry:
+- Show
+- Show label text
+- Right side
+- Show icon
+- Show text
+Position:
+- Shift left one plugin position on the bar
+- Shift right one plugin position on the bar
+- The shift is on the same bar
+- The shift will not move a plugin from one side to the other
+Bar:
+- Drop down so the user can pick the bar the plugin is to be shown on.
+- The list contains only the bars the user has selected to be shown.
+- The user can not move a plugin to a hidden bar to 'hide' it. The user should ensure "Show Plugin" is unchecked.
+:DESC
+VAR:  None
+OUT:  None
+--]]
+local function TitanUpdateConfigAddons()
+	local args = optionsAddons.args
+	local plug_in = nil
+	local plug_version = ""
+
+	wipe(args)
+
+	for idx, value in pairs(TitanPluginsIndex) do
+		plug_in = TitanUtils_GetPlugin(TitanPluginsIndex[idx])
+		if plug_in then
+			if plug_in.version then
+				plug_version = TitanUtils_GetGreenText(" (v"..plug_in.version..")")
+			else
+				plug_version = ""
+			end
+			args[plug_in.id] = {
+				type = "group",
+				name = (plug_in.menuText or ""),
+				order = idx,
+				args = {
+					name = {
+						type = "header",
+						name = ((plug_in.menuText or "")..plug_version),
+						order = 1,
+					},
+					show = {
+						type = "toggle",
+						name = L["TITAN_PANEL_MENU_SHOW"],
+						order = 3,
+						get = function(info) return (TitanPanel_IsPluginShown(info[1])) end,
+						set = function(info, v)
+							local name = info[1]
+							if v then -- Show / add
+								local bar = (TitanGetVar(name, "ForceBar") or TitanUtils_PickBar())
+								TitanUtils_AddButtonOnBar(bar, name)
+							else -- Hide / remove
+								TitanPanel_RemoveButton(name)
+							end
+							end,
+					},
+				}
+			}
+
+			--ShowIcon
+			if plug_in.controlVariables and plug_in.controlVariables.ShowIcon then
+				args[plug_in.id].args.icon =
+				{
+					type = "toggle",
+					name = L["TITAN_PANEL_MENU_SHOW_ICON"],
+					order = 4,
+					get = function(info) return (TitanGetVar(info[1], "ShowIcon")) end,
+					set = function(info, v)
+						TitanToggleVar(info[1], "ShowIcon");
+						TitanPanelButton_UpdateButton(info[1])
+						end,
+				}
+			end
+
+			--ShowLabel
+			if plug_in.controlVariables and plug_in.controlVariables.ShowLabelText then
+				args[plug_in.id].args.label = {
+					type = "toggle",
+					name = L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"],
+					order = 5,
+					get = function(info) return (TitanGetVar(info[1], "ShowLabelText")) end,
+					set = function(info, v)
+						TitanToggleVar(info[1], "ShowLabelText");
+						TitanPanelButton_UpdateButton(info[1])
+					end,
+				}
+			end
+
+			--ShowRegularText (LDB data sources only atm)
+			if plug_in.controlVariables and plug_in.controlVariables.ShowRegularText then
+				args[plug_in.id].args.regular_text =
+				{
+					type = "toggle",
+					name = L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"],
+					order = 6,
+					get = function(info) return (TitanGetVar(info[1], "ShowRegularText")) end,
+					set = function(info, v)
+						TitanToggleVar(info[1], "ShowRegularText");
+						TitanPanelButton_UpdateButton(info[1])
+						end,
+				}
+			end
+
+			--ShowColoredText
+			if plug_in.controlVariables and plug_in.controlVariables.ShowColoredText then
+				args[plug_in.id].args.color_text = {
+					type = "toggle",
+					name = L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"],
+					order = 7,
+					get = function(info) return (TitanGetVar(info[1], "ShowColoredText")) end,
+					set = function(info, v)
+						TitanToggleVar(info[1], "ShowColoredText");
+						TitanPanelButton_UpdateButton(info[1])
+					end,
+				}
+			end
+
+			-- Right-side plugin
+			if plug_in.controlVariables and plug_in.controlVariables.DisplayOnRightSide then
+				args[plug_in.id].args.right_side = {
+					type = "toggle",
+					name = L["TITAN_PANEL_MENU_LDB_SIDE"],
+					order = 8,
+					get = function(info) return (TitanGetVar(info[1], "DisplayOnRightSide")) end,
+					set = function(info, v)
+						local bar = TitanUtils_GetWhichBar(info[1])
+						TitanToggleVar(info[1], "DisplayOnRightSide");
+						TitanPanel_RemoveButton(info[1]);
+						TitanUtils_AddButtonOnBar(bar, info[1]);
+						TitanPanelButton_UpdateButton(info[1])
+					end,
+				}
+			end
+			-- Shift R / L
+			args[plug_in.id].args.plugin_position = {
+				order = 50,
+				type = "header",
+				name = L["TITAN_PANEL_MENU_POSITION"],
+			}
+			args[plug_in.id].args.shift_left = {
+				type = "execute",
+				name = "< "..L["TITAN_PANEL_SHIFT_LEFT"].."  ",
+				order = 51,
+				func = function(info, arg1)
+					local name = info[1]
+					if TitanPanel_IsPluginShown(name) then
+						TitanUtils_ShiftButtonOnBarLeft(name)
+					end
+				end,
+			}
+			args[plug_in.id].args.shift_right = {
+				type = "execute",
+				name = "> "..L["TITAN_PANEL_SHIFT_RIGHT"],
+				order = 52,
+				func = function(info, arg1)
+					local name = info[1]
+					if TitanPanel_IsPluginShown(info[1]) then
+						TitanUtils_ShiftButtonOnBarRight(name)
+					end
+				end,
+			}
+			args[plug_in.id].args.space_50_1 = {
+				order = 53,
+				type = "header",
+				name = L["TITAN_PANEL_MENU_BAR"],
+			}
+			if not TitanVarExists(plug_in.id, "ForceBar") then
+				args[plug_in.id].args.top_bottom = {
+					order = 54, type = "select",
+					name = L["TITAN_PANEL_MENU_BAR"],
+					desc = L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"],
+					get = function(info)
+						return TitanUtils_GetWhichBar(info[1]) end,
+					set = function(info,v)
+						local name = info[1]
+						if TitanPanel_IsPluginShown(name) then
+							TitanUtils_AddButtonOnBar(v, name)
+						end
+					end,
+					values = function()
+						local Locationlist = {}
+						local v
+						for idx,v in pairs (TitanBarData) do
+							if TitanPanelGetVar(TitanBarData[idx].name.."_Show") then
+								Locationlist[TitanBarData[idx].name] = TitanBarData[idx].locale_name
+							end
+						end
+						return Locationlist
+					end,
+				}
+			else
+				args[plug_in.id].args.top_bottom = {
+					order = 54,
+					type = "description",
+					name = TitanUtils_GetGoldText(L["TITAN_PANEL_MENU_BAR_ALWAYS"].." "..TitanGetVar(plug_in.id, "ForceBar")),
+					cmdHidden = true,
+				}
+			end
+		end
+	end
+
+	-- Config Tables changed!
+	AceConfigRegistry:NotifyChange("Titan Panel Addon Control")
+end
+-------------
+--[[ local
+NAME: optionsAdvanced
+DESC: Set the table to allow the user to control advanced features.
+Controls:
+- Entering world timer - some users need Titan to wait longer whenever the splash / loading screen is shown before adjusting frames and (re)setting data.
+- Vehicle timer - some users need Titan to wait longer whenever entering or leaving a vehicle before adjusting frames.
+:DESC
+--]]
+local optionsAdvanced = {
+	name = L["TITAN_PANEL_MENU_ADV"],
+	type = "group",
+	args = {
+		confdesc = {
+			order = 1,
+			type = "description",
+			name = L["TITAN_PANEL_MENU_ADV_DESC"],
+			cmdHidden = true
+			},
+		advtimerpew = {
+			name = L["TITAN_PANEL_MENU_ADV_PEW"],
+			desc = L["TITAN_PANEL_MENU_ADV_PEW_DESC"],
+			order = 10, type = "range", width = "full",
+			min = 1, max = 10, step = 0.5,
+			get = function() return TitanAllGetVar("TimerPEW") end,
+			set = function(_, a)
+				TitanAllSetVar("TimerPEW", a);
+				TitanTimers["EnterWorld"].delay = a
+			end,
+		},
+		advtimervehicle = {
+			name = L["TITAN_PANEL_MENU_ADV_VEHICLE"],
+			desc = L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"],
+			order = 50, type = "range", width = "full",
+			min = 1, max = 10, step = 0.5,
+			get = function() return TitanAllGetVar("TimerVehicle") end,
+			set = function(_, a)
+				TitanAllSetVar("TimerVehicle", a);
+				TitanTimers["Vehicle"].delay = a
+			end,
+		},
+	},
+}
+
+--[[ Titan
+NAME: TitanUpdateConfig
+DESC: This routine will handle the requests to update the various data items in Titan options screens.
+VAR:  None
+OUT:  None
+NOTE:
+- This is called after the plugins are registered in the 'player entering world' event. It can be called again as more plugins are registered.
+:NOTE
+--]]
+function TitanUpdateConfig(action)
+	if action == "init" then
+		-- Update the tables for the latest lists
+		TitanUpdateConfigAddons()
+		TitanUpdateAddonAttempts()
+		TitanUpdateExtras()
+		TitanUpdateChars()
+		TitanPanel_TransOptions(optionsTrans.args)
+	end
+	if action == "nuke" then
+		local nuked = {
+			name = "Titan could not initialize properly.", --L["TITAN_DEBUG"],
+			type = "group",
+			args = {}
+		}
+
+		TitanPrint("-- Clearing Titan options...", "warning")
+		-- Use the same group as below!!
+--		AceConfig:RegisterOptionsTable("Titan Panel Main", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Bars", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Aux Bars", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Frames", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Transparency Control", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Panel Control", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Skin Control", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Skin Custom", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Addon Control", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Addon Attempts", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Addon Extras", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Addon Chars", nuked)
+		AceConfig:RegisterOptionsTable("Titan Panel Addon Advanced", nuked)
+	end
+end
+
+--[[
+Register the options tables with Ace then register the options with Blizz so the user can use them.
+--]]
+-- Add Blizzard Configuration Panel
+AceConfig:RegisterOptionsTable("Titan Panel Main", optionsControl)
+AceConfig:RegisterOptionsTable("Titan Panel Bars", optionsBars)
+AceConfig:RegisterOptionsTable("Titan Panel Aux Bars", optionsAuxBars)
+AceConfig:RegisterOptionsTable("Titan Panel Frames", optionsFrames)
+AceConfig:RegisterOptionsTable("Titan Panel Transparency Control", optionsTrans)
+AceConfig:RegisterOptionsTable("Titan Panel Panel Control", optionsUIScale)
+AceConfig:RegisterOptionsTable("Titan Panel Skin Control", optionsSkins)
+AceConfig:RegisterOptionsTable("Titan Panel Skin Custom", optionsSkinsCustom)
+AceConfig:RegisterOptionsTable("Titan Panel Addon Control", optionsAddons)
+AceConfig:RegisterOptionsTable("Titan Panel Addon Attempts", optionsAddonAttempts)
+AceConfig:RegisterOptionsTable("Titan Panel Addon Extras", optionsExtras)
+AceConfig:RegisterOptionsTable("Titan Panel Addon Chars", optionsChars)
+AceConfig:RegisterOptionsTable("Titan Panel Addon Advanced", optionsAdvanced)
+-- Set the main options pages
+AceConfigDialog:AddToBlizOptions("Titan Panel Main", L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Bars", L["TITAN_PANEL_MENU_TOP_BARS"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Aux Bars", L["TITAN_PANEL_MENU_BOTTOM_BARS"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Addon Control", L["TITAN_PANEL_MENU_PLUGINS"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Addon Chars", L["TITAN_PANEL_MENU_PROFILES"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Frames", L["TITAN_PANEL_MENU_OPTIONS_SHORT"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Panel Control", L["TITAN_UISCALE_MENU_TEXT_SHORT"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Transparency Control", L["TITAN_TRANS_MENU_TEXT_SHORT"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Skin Control", L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Skin Custom", L["TITAN_SKINS_OPTIONS_CUSTOM"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Addon Extras", L["TITAN_PANEL_EXTRAS_SHORT"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Addon Attempts", L["TITAN_PANEL_ATTEMPTS_SHORT"], L["TITAN_PANEL"])
+AceConfigDialog:AddToBlizOptions("Titan Panel Addon Advanced", L["TITAN_PANEL_MENU_ADV"], L["TITAN_PANEL"])
+
diff --git a/Titan/TitanPanel.lua b/Titan/TitanPanel.lua
new file mode 100644
index 0000000..22267ba
--- /dev/null
+++ b/Titan/TitanPanel.lua
@@ -0,0 +1,2154 @@
+--[[ File
+NAME: TitanPanel.lua
+DESC: Contains the basic routines of Titan. All the event handler routines, initialization routines, Titan menu routines, and select plugin handler routines.
+--]]
+
+-- Locals
+local TPC = TITAN_PANEL_CONSTANTS -- shortcut
+local TITAN_PANEL_BUTTONS_INIT_FLAG = nil;
+
+local TITAN_PANEL_FROM_TOP = -25;
+local TITAN_PANEL_FROM_BOTTOM = 25;
+local TITAN_PANEL_FROM_BOTTOM_MAIN = 1;
+local TITAN_PANEL_FROM_TOP_MAIN = 1;
+
+local _G = getfenv(0);
+local InCombatLockdown = _G.InCombatLockdown;
+local IsTitanPanelReset = nil;
+local numOfTextures = 0;
+local numOfTexturesHider = 0;
+
+-- Library references
+local L = LibStub("AceLocale-3.0"):GetLocale(TITAN_ID, true)
+local AceTimer = LibStub("AceTimer-3.0")
+local media = LibStub("LibSharedMedia-3.0")
+
+--------------------------------------------------------------
+--
+function TitanPanel_OkToReload()
+	StaticPopupDialogs["TITAN_RESET_RELOAD"] = {
+		text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"])
+			.."\n\n"..L["TITAN_PANEL_RESET_WARNING"],
+		button1 = ACCEPT,
+		button2 = CANCEL,
+		OnAccept = function(self)
+			ReloadUI()
+		end,
+		showAlert = 1,
+		timeout = 0,
+		whileDead = 1,
+		hideOnEscape = 1
+	};
+	StaticPopup_Show("TITAN_RESET_RELOAD");
+end
+
+--[[ Titan
+NAME: TitanPanel_ResetToDefault
+DESC: Give the user a 'are you sure'. If the user accepts then reset current toon back to default Titan settings.
+VAR:  None
+OUT:  None
+NOTE:
+- Even if the user was using global profiles they will not when this is done.
+:NOTE
+--]]
+function TitanPanel_ResetToDefault()
+	StaticPopupDialogs["TITAN_RESET_BAR"] = {
+		text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"])
+			.."\n\n"..L["TITAN_PANEL_RESET_WARNING"],
+		button1 = ACCEPT,
+		button2 = CANCEL,
+		OnAccept = function(self)
+			TitanVariables_UseSettings(TitanSettings.Player, TITAN_PROFILE_RESET);
+			IsTitanPanelReset = true;
+--TitanPanel_OkToReload()
+			ReloadUI()
+		end,
+		showAlert = 1,
+		timeout = 0,
+		whileDead = 1,
+		hideOnEscape = 1
+	};
+	StaticPopup_Show("TITAN_RESET_BAR");
+end
+
+--[[ Titan
+NAME: TitanPanel_SaveCustomProfile
+DESC: The user wants to save a custom Titan profile. Show the user the dialog boxes to make it happen.
+VAR:  None
+OUT:  None
+NOTE:
+- The profile is written to the Titan saved variables. A reload of the UI is needed to ensure the profile is written to disk for the user to load later.
+:NOTE
+--]]
+function TitanPanel_SaveCustomProfile()
+   -- Create the dialog box code we'll need...
+
+	-- helper to actually write the profile to the Titan saved vars
+	local function Write_profile(name)
+		local currentprofilevalue, _, _ = TitanUtils_GetPlayer()
+		local profileName = TitanUtils_CreateName(name, TITAN_CUSTOM_PROFILE_POSTFIX)
+		TitanSettings.Players[profileName] =
+			TitanSettings.Players[currentprofilevalue]
+		TitanPrint(L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"]
+			.."'"..name.."'"
+			, "info")
+	end
+	-- helper to ask the user to overwrite a profile
+	local function Overwrite_profile(name)
+		local dialogFrame =
+			StaticPopup_Show("TITAN_OVERWRITE_CUSTOM_PROFILE", name);
+		if dialogFrame then
+			dialogFrame.data = name;
+		end
+	end
+	-- helper to handle getting the profile name from the user
+	local function Get_profile_name(self)
+		local rawprofileName = self.editBox:GetText();
+		-- remove any spaces the user may have typed in the name
+		local conc2profileName = string.gsub( rawprofileName, " ", "" );
+		if conc2profileName == "" then return; end
+		-- no '@' is allowed or it will mess with the Titan profile naming convention
+		local concprofileName = string.gsub( conc2profileName, TITAN_AT, "-" );
+		local profileName = TitanUtils_CreateName(concprofileName, TITAN_CUSTOM_PROFILE_POSTFIX)
+		if TitanSettings.Players[profileName] then
+			-- Warn the user of an existing profile
+			Overwrite_profile(rawprofileName)
+			self:Hide();
+			return;
+		else
+			-- Save the requested profile
+			Write_profile(rawprofileName)
+			self:Hide();
+			StaticPopup_Show("TITAN_RELOADUI");
+		end
+	end
+	-- Dialog box to warn the user that the UI will be reloaded
+	-- This ensures the profile is written to disk
+	StaticPopupDialogs["TITAN_RELOADUI"] = {
+		text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"]).."\n\n"
+			..L["TITAN_PANEL_MENU_PROFILE_RELOADUI"],
+		button1 = "OKAY",
+		OnAccept = function(self)
+			ReloadUI(); -- ensure profile is written to disk
+		end,
+		showAlert = 1,
+		whileDead = 1,
+		timeout = 0,
+	};
+
+	-- Dialog box to warn the user that an existing profile will be overwritten.
+	StaticPopupDialogs["TITAN_OVERWRITE_CUSTOM_PROFILE"] = {
+		text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"]).."\n\n"
+			..L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"],
+		button1 = ACCEPT,
+		button2 = CANCEL,
+		OnAccept = function(self, data)
+			Write_profile(data)
+			self:Hide();
+			StaticPopup_Show("TITAN_RELOADUI");
+		end,
+		showAlert = 1,
+		whileDead = 1,
+		timeout = 0,
+		hideOnEscape = 1
+	};
+
+	-- Dialog box to save the profile.
+	StaticPopupDialogs["TITAN_SAVE_CUSTOM_PROFILE"] = {
+		text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"]).."\n\n"
+			..L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"],
+		button1 = ACCEPT,
+		button2 = CANCEL,
+		hasEditBox = 1,
+		maxLetters = 20,
+		OnAccept = function(self)
+			-- self refers to this frame with the Accept button
+			Get_profile_name(self)
+		end,
+		OnShow = function(self)
+			self.editBox:SetFocus();
+		end,
+		OnHide = function(self)
+			self.editBox:SetText("");
+		end,
+		EditBoxOnEnterPressed = function(self)
+			-- We need to get the parent because self refers to the edit box.
+			Get_profile_name(self:GetParent())
+			end,
+		EditBoxOnEscapePressed = function(self)
+			self:GetParent():Hide();
+		end,
+		timeout = 0,
+		exclusive = 1,
+		whileDead = 1,
+		hideOnEscape = 1
+	};
+
+	StaticPopup_Show("TITAN_SAVE_CUSTOM_PROFILE");
+
+	-- Can NOT cleanup. Execution does not stop when a dialog box is invoked!
+--	StaticPopupDialogs["TITAN_RELOADUI"] = {}
+--	StaticPopupDialogs["TITAN_OVERWRITE_CUSTOM_PROFILE"] = {}
+--	StaticPopupDialogs["TITAN_SAVE_CUSTOM_PROFILE"] = {}
+
+end
+
+--[[ Titan
+NAME: TitanSetPanelFont
+DESC: Set or change the font and font size of text on the Titan bar. This affects ALL plugins.
+VAR: fontname - The text name of the font to use. Defaults to Titan default if none given.
+VAR: fontsize - The size of the font to use. Defaults to Titan default if none given.
+OUT: None
+NOTE:
+- Each registered plugin will have its font updated. Then all plugins will be refreshed to show the new font.
+:NOTE
+--]]
+function TitanSetPanelFont(fontname, fontsize)
+	-- a couple of arg checks to avoid unpleasant things...
+	if not fontname then fontname = TPC.FONT_NAME end
+	if not fontsize then fontsize = TPC.FONT_SIZE end
+	local index,id;
+	local newfont = media:Fetch("font", fontname)
+	for index, id in pairs(TitanPluginsIndex) do
+		local button = TitanUtils_GetButton(id);
+		local buttonText = _G[button:GetName()..TITAN_PANEL_TEXT];
+		if buttonText then
+			buttonText:SetFont(newfont, fontsize);
+		end
+		-- account for plugins with child buttons
+		local childbuttons = {button:GetChildren()};
+		for _, child in ipairs(childbuttons) do
+			if child then
+				local bname = _G[child:GetName()]
+				if bname then
+					local childbuttonText = _G[child:GetName()..TITAN_PANEL_TEXT];
+					if childbuttonText then
+						childbuttonText:SetFont(newfont, fontsize);
+					end
+				end
+			end
+		end
+	end
+	TitanPanel_RefreshPanelButtons();
+end
+
+
+--[[ not_used
+local function TitanPanel_SetTransparent(frame, position)
+	local frName = _G[frame];
+
+	if (position == TITAN_PANEL_PLACE_TOP) then
+		frName:ClearAllPoints();
+		frName:SetPoint("TOPLEFT", "UIParent", "TOPLEFT", 0, 0);
+		frName:SetPoint("BOTTOMRIGHT", "UIParent", "TOPRIGHT", 0, -TITAN_PANEL_BAR_HEIGHT);
+	else
+		frName:ClearAllPoints();
+		frName:SetPoint("BOTTOMLEFT", "UIParent", "BOTTOMLEFT", 0, 0);
+		frName:SetPoint("TOPRIGHT", "UIParent", "BOTTOMRIGHT", 0, TITAN_PANEL_BAR_HEIGHT);
+	end
+end
+--]]
+
+--[[ local
+NAME: TitanPanel_CreateABar
+DESC: Helper to create the Titan bar passed in.
+VAR: frame - The frame name (string) of the Titan bar to create
+OUT: None
+NOTE:
+- This also creates the hider bar in case the user want to use auto hide.
+:NOTE
+--]]
+local function TitanPanel_CreateABar(frame)
+	if not frame then return end
+	local hide_name = TitanBarData[frame].hider
+	local bar_name = TitanBarData[frame].name
+
+	if hide_name and bar_name then
+		-- Set script handlers for display
+		_G[frame]:RegisterForClicks("LeftButtonUp", "RightButtonUp");
+		_G[frame]:SetScript("OnEnter", function(self) TitanPanelBarButton_OnEnter(self) end)
+		_G[frame]:SetScript("OnLeave", function(self) TitanPanelBarButton_OnLeave(self) end)
+		_G[frame]:SetScript("OnClick", function(self, button) TitanPanelBarButton_OnClick(self, button) end)
+
+		-- Set script handlers for display
+		_G[hide_name]:RegisterForClicks("LeftButtonUp", "RightButtonUp");
+		_G[hide_name]:SetScript("OnEnter", function(self) TitanPanelBarButtonHider_OnEnter(self) end)
+		_G[hide_name]:SetScript("OnLeave", function(self) TitanPanelBarButtonHider_OnLeave(self) end)
+		_G[hide_name]:SetScript("OnClick", function(self, button) TitanPanelBarButton_OnClick(self, button) end)
+
+		_G[hide_name]:SetFrameStrata("BACKGROUND")
+		_G[hide_name]:SetHeight(TITAN_PANEL_BAR_HEIGHT/2);
+		_G[hide_name]:SetWidth(2560);
+
+		-- Set the display bar
+		local container = _G[frame]
+		container:SetHeight(TITAN_PANEL_BAR_HEIGHT);
+		-- Set local identifier
+		local container_text = _G[frame.."_Text"]
+		if container_text then -- was used for debug/creating of the independent bars
+			container_text:SetText(tostring(bar_name))
+			-- for now show it
+			container:Show()
+		end
+	end
+end
+
+--------------------------------------------------------------
+--
+-- Event registration
+--_G[TITAN_PANEL_CONTROL]:RegisterEvent("ADDON_LOADED");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("PLAYER_ENTERING_WORLD");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("PLAYER_REGEN_DISABLED");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("PLAYER_REGEN_ENABLED");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("CVAR_UPDATE");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("PLAYER_LOGOUT");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("ACTIVE_TALENT_GROUP_CHANGED");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("UNIT_ENTERED_VEHICLE");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("UNIT_EXITED_VEHICLE");
+_G[TITAN_PANEL_CONTROL]:SetScript("OnEvent", function(_, event, ...)
+	_G[TITAN_PANEL_CONTROL][event](_G[TITAN_PANEL_CONTROL], ...)
+end)
+-- For the pet battle - for now we'll hide the Titan bars...
+-- Cannot seem to move the 'top' part of the frame.
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("PET_BATTLE_OPENING_START");
+_G[TITAN_PANEL_CONTROL]:RegisterEvent("PET_BATTLE_CLOSE");
+
+
+--[[ Titan
+NAME: TitanPanel_PlayerEnteringWorld
+DESC: Do all the setup needed when a user logs in / reload UI / enter or leave an instance.
+VAR:  None
+OUT:  None
+NOTE:
+- This is called after the 'player entering world' event is fired by Blizz.
+- This is also used when a LDB plugin is created after Titan runs the 'player entering world' code.
+:NOTE
+--]]
+function TitanPanel_PlayerEnteringWorld()
+	if Titan__InitializedPEW then
+		-- Also sync the LDB object with the Tian plugin
+		TitanLDBRefreshButton()
+	else
+		-- Get Profile and Saved Vars
+		TitanVariables_InitTitanSettings();
+
+		-- only do this sort of initialization on the first PEW event
+		if not TitanAllGetVar("Silenced") then
+			TitanPrint("", "header")
+		end
+
+		if not ServerTimeOffsets then
+			ServerTimeOffsets = {};
+		end
+		if not ServerHourFormat then
+			ServerHourFormat = {};
+		end
+
+		-- Set the two anchors in their default positions
+		-- until the Titan bars are drawn
+		TitanPanelTopAnchor:ClearAllPoints();
+		TitanPanelTopAnchor:SetPoint("TOPLEFT", "UIParent", "TOPLEFT", 0, 0);
+		TitanPanelBottomAnchor:ClearAllPoints();
+		TitanPanelBottomAnchor:SetPoint("BOTTOMLEFT", "UIParent", "BOTTOMLEFT", 0, 0);
+
+		-- Ensure the bars are created before the
+		-- plugins are registered.
+		for idx, v in pairs (TitanBarData) do
+			TitanPanel_CreateABar(idx)
+		end
+
+		local realmName = GetRealmName()
+
+		if ServerTimeOffsets[realmName] then
+			TitanSetVar(TITAN_CLOCK_ID, "OffsetHour", ServerTimeOffsets[realmName])
+		elseif TitanGetVar(TITAN_CLOCK_ID, "OffsetHour") then
+			ServerTimeOffsets[realmName] = TitanGetVar(TITAN_CLOCK_ID, "OffsetHour")
+		end
+
+		if ServerHourFormat[realmName] then
+			TitanSetVar(TITAN_CLOCK_ID, "Format", ServerHourFormat[realmName])
+		elseif TitanGetVar(TITAN_CLOCK_ID, "Format") then
+			ServerHourFormat[realmName] = TitanGetVar(TITAN_CLOCK_ID, "Format")
+		end
+
+		-- Check to see if we should kill off the OrderHallCommandBar
+		if not TitanAllGetVar("OrderHall") then
+			local TitanPanelAce = LibStub("AceAddon-3.0"):NewAddon("TitanPanelOHCB", "AceHook-3.0")
+			TitanPanelAce:SecureHook("OrderHall_CheckCommandBar",
+				function()
+					if OrderHallCommandBar then
+						OrderHallCommandBar:Hide()
+						OrderHallCommandBar:UnregisterAllEvents()
+						OrderHallCommandBar.Show = function () end
+					end
+				end
+			)
+		else
+			local TitanPanelAce = LibStub("AceAddon-3.0"):NewAddon("TitanPanelOHCB", "AceHook-3.0")
+			TitanPanelAce:Unhook("OrderHall_CheckCommandBar")
+		end
+	end
+	local _ = nil
+	TitanSettings.Player,_,_ = TitanUtils_GetPlayer()
+	-- Some addons wait to create their LDB component or a Titan addon could
+	-- create additional buttons as needed.
+	-- So we need to sync their variables and set them up
+	TitanUtils_RegisterPluginList()
+
+	-- Init detailed settings only after plugins are registered!
+	TitanVariables_UseSettings(nil, TITAN_PROFILE_INIT)
+
+	-- all addons are loaded so update the config (options)
+	-- some could have registered late...
+	TitanUpdateConfig("init")
+
+	-- Init panel font
+	local isfontvalid = media:IsValid("font", TitanPanelGetVar("FontName"))
+	if isfontvalid then
+		TitanSetPanelFont(TitanPanelGetVar("FontName"), TitanPanelGetVar("FontSize"))
+	else
+	-- if the selected font is not valid, revert to default (Friz Quadrata TT)
+		TitanPanelSetVar("FontName", TPC.FONT_NAME);
+		TitanSetPanelFont(TPC.FONT_NAME, TitanPanelGetVar("FontSize"))
+	end
+
+	-- Init panel frame strata
+	TitanVariables_SetPanelStrata(TitanPanelGetVar("FrameStrata"))
+
+	-- Titan Panel has initialized its variables and registered plugins.
+	-- Allow Titan - and others - to adjust the bars
+	Titan__InitializedPEW = true
+
+	-- Move frames
+	TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_BOTH, true)
+	TitanMovable_SecureFrames()
+
+	-- Secondary failsafe check for bottom frame adjustment
+	--
+	-- On longer game loads (log in, reload, instances, etc Titan will adjust
+	-- then Blizz will adjust putting the action buttons over / under Titan
+	-- if the user has aux 1/2 shown.
+	TitanMovable_AdjustTimer("EnterWorld")
+
+end
+
+--------------------------------------------------------------
+--
+-- Event handlers
+--
+function TitanPanelBarButton:ADDON_LOADED(addon)
+	if addon == TITAN_ID then
+		-- Get Profile and Saved Vars
+		TitanVariables_InitTitanSettings();
+		local VERSION = TitanPanel_GetVersion();
+		local POS = strfind(VERSION," - ");
+		VERSION = strsub(VERSION,1,POS-1);
+		TitanPrint("", "header")
+
+		if not ServerTimeOffsets then
+			ServerTimeOffsets = {};
+		end
+		if not ServerHourFormat then
+			ServerHourFormat = {};
+		end
+		-- Unregister event - saves a few event calls.
+		self:UnregisterEvent("ADDON_LOADED");
+		self.ADDON_LOADED = nil
+	end
+end
+
+function TitanPanelBarButton:PLAYER_ENTERING_WORLD()
+	local call_success = nil
+	local ret_val = nil
+
+	call_success, -- needed for pcall
+	ret_val =  -- actual return values
+		pcall (TitanPanel_PlayerEnteringWorld)
+	-- pcall does not allow errors to propagate out. Any error
+	-- is returned as text with the success / fail.
+	-- Think of it as sort of a try - catch block
+	if call_success then
+		-- Titan initialized properly
+	else
+		-- something really bad occured...
+		TitanPrint("Titan could not initialize!!!!  Cleaning up...", "error")
+		TitanPrint("--"..ret_val, "error")
+		-- Clean up best we can and tell the user to submit a ticket.
+		-- This could be the 1st log in or a reload (reload, instance, boat, ...)
+
+		-- Hide the bars. At times they are there but at 0% transparency.
+		-- They can be over the Blizz action bars creating havoc.
+		TitanPrint("-- Hiding Titan bars...", "warning")
+		TitanPanelBarButton_HideAllBars()
+
+		-- Remove the options pages
+		TitanUpdateConfig("nuke")
+		-- What else to clean up???
+
+		-- raise the error to WoW for display, if display errors is set.
+		-- This *must be* the last statement of the routine!
+		error(ret_val, 1)
+	end
+end
+
+function TitanPanelBarButton:CVAR_UPDATE(cvarname, cvarvalue)
+	if cvarname == "USE_UISCALE"
+	or cvarname == "WINDOWED_MODE"
+	or cvarname == "uiScale" then
+		if TitanPlayerSettings and TitanPanelGetVar("Scale") then
+			Titan_AdjustScale()
+			-- Adjust frame positions
+			TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_BOTH, true)
+		end
+	end
+end
+
+function TitanPanelBarButton:PLAYER_LOGOUT()
+	if not IsTitanPanelReset then
+		-- for debug
+		if TitanPanelRegister then
+			TitanPanelRegister.ToBe = TitanPluginToBeRegistered
+			TitanPanelRegister.ToBeNum = TitanPluginToBeRegisteredNum
+			TitanPanelRegister.TitanPlugins = TitanPlugins
+			TitanPanelRegister.Extras = TitanPluginExtras
+		end
+	end
+	Titan__InitializedPEW = nil
+end
+
+function TitanPanelBarButton:PLAYER_REGEN_DISABLED()
+	-- If in combat close all control frames and menus
+	TitanUtils_CloseAllControlFrames();
+	TitanUtils_CloseRightClickMenu();
+end
+
+function TitanPanelBarButton:PLAYER_REGEN_ENABLED()
+	-- Outside combat check to see if frames need correction
+	TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_BOTH, true)
+end
+
+function TitanPanelBarButton:ACTIVE_TALENT_GROUP_CHANGED()
+	TitanMovable_AdjustTimer("DualSpec")
+end
+
+function TitanPanelBarButton:UNIT_ENTERED_VEHICLE(self, ...)
+	TitanMovable_AdjustTimer("Vehicle")
+end
+function TitanPanelBarButton:UNIT_EXITED_VEHICLE(self, ...)
+	TitanMovable_AdjustTimer("Vehicle")
+end
+--
+function TitanPanelBarButton:PET_BATTLE_OPENING_START()
+--	TitanDebug("Pet_battle start: ")
+	-- Hide all bars and hiders
+	TitanPanelBarButton_HideAllBars()
+end
+function TitanPanelBarButton:PET_BATTLE_CLOSE()
+--	TitanDebug("Pet_battle end: ")
+	-- Show the bars the user had selected
+	TitanPanelBarButton_DisplayBarsWanted()
+end
+--
+--
+
+--[[ Titan
+NAME: TitanPanelBarButton_OnClick
+DESC: Handle the button clicks on any Titan bar.
+VAR: self - expected to be a Titan bar
+VAR: button - which mouse button was clicked
+OUT:  None
+NOTE:
+- This only reacts to the right or left mouse click without modifiers.
+- Used in the set script for the Titan display and hider frames
+:NOTE
+--]]
+function TitanPanelBarButton_OnClick(self, button)
+	-- ensure that the right-click menu will not appear on "hidden" bottom bar(s)
+	local bar = self:GetName();
+	if (button == "LeftButton") then
+		TitanUtils_CloseAllControlFrames();
+		TitanUtils_CloseRightClickMenu();
+	elseif (button == "RightButton") then
+		TitanUtils_CloseAllControlFrames();
+		TitanPanelRightClickMenu_Close();
+		-- Show RightClickMenu anyway
+		TitanPanelDisplayRightClickMenu_Toggle(self)  -- self not used...
+	end
+end
+
+--
+-- Slash command handler
+--
+--[[ local
+NAME: TitanPanel_ParseSlashCmd
+DESC: Helper to parse the user commands.
+VAR: cmd - user string from the command 'window'
+OUT: table - table of 'words' the user typed in
+NOTE:
+- each 'word' in words table is made lower case for comparison simplicity
+:NOTE
+--]]
+local function TitanPanel_ParseSlashCmd(cmd)
+	local words = {}
+	for w in string.gmatch (cmd, "%w+") do
+		words [#words + 1] = (w and string.lower(w) or "?")
+	end
+--[[
+	local tmp = ""
+	for idx,v in pairs (words) do
+		tmp = tmp.."'"..words[idx].."' "
+	end
+
+	TitanDebug (tmp.." : "..#words)
+--]]
+	return words
+end
+
+--[[ local
+NAME: handle_slash_help
+DESC: Helper to tell the user the relevant Titan commands.
+VAR:  cmd - string 'all' | 'reset' | 'gui' | 'silent'
+OUT:  None
+NOTE:
+- Depending on cmd put to chat the appropriate help
+:NOTE
+--]]
+local function handle_slash_help(cmd)
+	cmd = cmd or "all"
+
+	--	Give the user the general help if we can not figure out what they want
+	TitanPrint("", "header")
+	-- Cannot count registered plugins after initial registration  TitanUtils_RegisterPluginList()
+
+	if cmd == "reset" then
+		TitanPrint(L["TITAN_PANEL_SLASH_RESET_0"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_RESET_1"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_RESET_2"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_RESET_3"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_RESET_4"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_RESET_5"], "plain")
+	end
+	if cmd == "gui" then
+		TitanPrint(L["TITAN_PANEL_SLASH_GUI_0"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_GUI_1"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_GUI_2"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_GUI_3"], "plain")
+	end
+	if cmd == "profile" then
+		TitanPrint(L["TITAN_PANEL_SLASH_PROFILE_0"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_PROFILE_1"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_PROFILE_2"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_PROFILE_3"], "plain")
+	end
+	if cmd == "silent" then
+		TitanPrint(L["TITAN_PANEL_SLASH_SILENT_0"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_SILENT_1"], "plain")
+	end
+	if cmd == "orderhall" then
+		TitanPrint(L["TITAN_PANEL_SLASH_ORDERHALL_0"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_ORDERHALL_1"], "plain")
+	end
+	if cmd == "help" then
+		TitanPrint(L["TITAN_PANEL_SLASH_HELP_0"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_HELP_1"], "plain")
+	end
+	if cmd == "all" then
+		TitanPrint(L["TITAN_PANEL_SLASH_ALL_0"], "plain")
+		TitanPrint(L["TITAN_PANEL_SLASH_ALL_1"], "plain")
+	end
+end
+
+--[[ local
+NAME: handle_reset_cmds
+DESC: Helper to execute the various reset commands from the user.
+VAR:  cmd_list - A table containing the list of 'words' the user typed in
+OUT:  None
+--]]
+local function handle_reset_cmds(cmd_list)
+	local cmd = cmd_list[1]
+	local p1 = cmd_list[2] or nil
+	-- sanity check
+	if (not cmd == "reset") then
+		return
+	end
+
+	if p1 == nil then
+		TitanPanel_ResetToDefault();
+	elseif p1 == "tipfont" then
+		TitanPanelSetVar("TooltipFont", 1);
+		GameTooltip:SetScale(TitanPanelGetVar("TooltipFont"));
+		TitanPrint(L["TITAN_PANEL_SLASH_RESP1"], "info")
+	elseif p1 == "tipalpha" then
+		TitanPanelSetVar("TooltipTrans", 1);
+		local red, green, blue, _ = GameTooltip:GetBackdropColor();
+		local red2, green2, blue2, _ = GameTooltip:GetBackdropBorderColor();
+		GameTooltip:SetBackdropColor(red,green,blue,TitanPanelGetVar("TooltipTrans"));
+		GameTooltip:SetBackdropBorderColor(red2,green2,blue2,TitanPanelGetVar("TooltipTrans"));
+		TitanPrint(L["TITAN_PANEL_SLASH_RESP2"], "info")
+	elseif p1 == "panelscale" then
+		if not InCombatLockdown() then
+			TitanPanelSetVar("Scale", 1);
+			-- Adjust panel scale
+			Titan_AdjustScale()
+			-- Adjust frame positions
+			TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_BOTH, true)
+			TitanPrint(L["TITAN_PANEL_SLASH_RESP3"], "info")
+		else
+			TitanPrint(L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"], "warning")
+		end
+	elseif p1 == "spacing" then
+		TitanPanelSetVar("ButtonSpacing", 20);
+		TitanPanel_InitPanelButtons();
+		TitanPrint(L["TITAN_PANEL_SLASH_RESP4"], "info")
+	else
+		handle_slash_help("reset")
+	end
+end
+
+--[[ local
+NAME: handle_giu_cmds
+DESC: Helper to execute the gui related commands from the user.
+VAR: cmd_list - A table containing the list of 'words' the user typed in
+OUT: None
+--]]
+local function handle_giu_cmds(cmd_list)
+	local cmd = cmd_list[1]
+	local p1 = cmd_list[2] or nil
+	-- sanity check
+	if (not cmd == "gui") then
+		return
+	end
+
+	if p1 == "control" then
+		InterfaceOptionsFrame_OpenToCategory(L["TITAN_UISCALE_MENU_TEXT_SHORT"])
+	elseif p1 == "trans" then
+		InterfaceOptionsFrame_OpenToCategory(L["TITAN_TRANS_MENU_TEXT_SHORT"])
+	elseif p1 == "skin" then
+		InterfaceOptionsFrame_OpenToCategory(L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"])
+	else
+		handle_slash_help("gui")
+	end
+end
+
+--[[ local
+NAME: handle_profile_cmds
+DESC: Helper to execute the profile related commands from the user.
+VAR: cmd_list - A table containing the list of 'words' the user typed in
+OUT: None
+--]]
+local function handle_profile_cmds(cmd_list)
+	local cmd = cmd_list[1]
+	local p1 = cmd_list[2] or nil
+	local p2 = cmd_list[3] or nil
+	local p3 = cmd_list[4] or nil
+	-- sanity check
+	if (not cmd == "profile") then
+		return
+	end
+
+	if p1 == "use" and p2 and p3 then
+		if TitanAllGetVar("GlobalProfileUse") then
+			TitanPrint(L["TITAN_PANEL_GLOBAL_ERR_1"], "info")
+		else
+			TitanVariables_UseSettings(TitanUtils_CreateName(p2, p3), TITAN_PROFILE_USE)
+		end
+	else
+		handle_slash_help("profile")
+	end
+end
+
+--[[ local
+NAME: handle_silent_cmds
+DESC: Helper to execute the silent commands from the user.
+VAR: cmd_list - A table containing the list of 'words' the user typed in
+OUT: None
+--]]
+local function handle_silent_cmds(cmd_list)
+	local cmd = cmd_list[1]
+	local p1 = cmd_list[2] or nil
+	-- sanity check
+	if (not cmd == "silent") then
+		return
+	end
+
+	if TitanAllGetVar("Silenced") then
+		TitanAllSetVar("Silenced", false);
+		TitanPrint(L["TITAN_PANEL_MENU_SILENT_LOAD"].." ".. L["TITAN_PANEL_MENU_DISABLED"], "info")
+	else
+		TitanAllSetVar("Silenced", true);
+		TitanPrint(L["TITAN_PANEL_MENU_SILENT_LOAD"].." ".. L["TITAN_PANEL_MENU_ENABLED"], "info")
+	end
+end
+
+--[[ local
+NAME: handle_orderhall_cmds
+DESC: Helper to execute the orderhall commands from the user.
+VAR: cmd_list - A table containing the list of 'words' the user typed in
+OUT: None
+--]]
+local function handle_orderhall_cmds(cmd_list)
+	local cmd = cmd_list[1]
+	local p1 = cmd_list[2] or nil
+	-- sanity check
+	if (not cmd == "orderhall") then
+		return
+	end
+
+	if TitanAllGetVar("OrderHall") then
+		TitanAllSetVar("OrderHall", false);
+		TitanPrint(L["TITAN_PANEL_MENU_HIDE_ORDERHALL"].." ".. L["TITAN_PANEL_MENU_ENABLED"], "info")
+		StaticPopupDialogs["TITAN_RELOAD"] = {
+			text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"]).."\n\n"
+				..L["TITAN_PANEL_RELOAD"],
+			button1 = ACCEPT,
+			button2 = CANCEL,
+			OnAccept = function(self)
+				ReloadUI();
+				end,
+			showAlert = 1,
+			timeout = 0,
+			whileDead = 1,
+			hideOnEscape = 1
+		};
+		StaticPopup_Show("TITAN_RELOAD");
+	else
+		TitanAllSetVar("OrderHall", true);
+		TitanPrint(L["TITAN_PANEL_MENU_HIDE_ORDERHALL"].." ".. L["TITAN_PANEL_MENU_DISABLED"], "info")
+		StaticPopupDialogs["TITAN_RELOAD"] = {
+			text = TitanUtils_GetNormalText(L["TITAN_PANEL_MENU_TITLE"]).."\n\n"
+				..L["TITAN_PANEL_RELOAD"],
+			button1 = ACCEPT,
+			button2 = CANCEL,
+			OnAccept = function(self)
+				ReloadUI();
+				end,
+			showAlert = 1,
+			timeout = 0,
+			whileDead = 1,
+			hideOnEscape = 1
+		};
+		StaticPopup_Show("TITAN_RELOAD");
+	end
+end
+
+--[[ local
+NAME: handle_help_cmds
+DESC: Helper to execute the help commands from the user.
+VAR: cmd_list - A table containing the list of 'words' the user typed in
+OUT: None
+--]]
+local function handle_help_cmds(cmd_list)
+	local cmd = cmd_list[1]
+	local p1 = cmd_list[2] or nil
+	-- sanity check
+	if (not cmd == "help") then
+		return
+	end
+
+	handle_slash_help(p1 or "all")
+end
+
+--[[ local
+NAME: TitanPanel_RegisterSlashCmd
+DESC: Helper to parse and execute all the Titan slash commands from the user.
+VAR: cmd - The command (string) the user typed in
+OUT: None
+--]]
+local function TitanPanel_RegisterSlashCmd(cmd_str)
+	local cmd_list = {}
+	-- parse what the user typed
+	cmd_list = TitanPanel_ParseSlashCmd(cmd_str)
+	local cmd = cmd_list[1] or ""
+	local p1 = cmd_list[2] or ""
+	local p2 = cmd_list[3] or ""
+	local p3 = cmd_list[4] or ""
+
+--	TitanDebug (cmd.." : "..p1.." "..p2.." "..p3.." "..#cmd_list)
+
+	if (cmd == "reset") then
+		handle_reset_cmds(cmd_list)
+	elseif (cmd == "gui") then
+		handle_giu_cmds(cmd_list)
+	elseif (cmd == "profile") then
+		handle_profile_cmds(cmd_list)
+	elseif (cmd == "silent") then
+		handle_silent_cmds(p1)
+	elseif (cmd == "orderhall") then
+		handle_orderhall_cmds(p1)
+	elseif (cmd == "help") then
+		handle_slash_help(p1)
+	else
+		handle_slash_help("all")
+	end
+end
+
+--------------------------------------------------------------
+--
+-- Register slash commands for Titan Panel
+SlashCmdList["TitanPanel"] = TitanPanel_RegisterSlashCmd;
+SLASH_TitanPanel1 = "/titanpanel";
+SLASH_TitanPanel2 = "/titan";
+
+--------------------------------------------------------------
+--
+-- Texture routines
+--[[ Titan
+NAME: TitanPanel_ClearAllBarTextures
+DESC: Clear the current texture from all Titan bars.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanel_ClearAllBarTextures()
+	-- Clear textures if they already exist
+	for idx,v in pairs (TitanBarData) do
+		for i = 0, numOfTexturesHider do
+			tex = TITAN_PANEL_BACKGROUND_PREFIX..TitanBarData[idx].name.."_"..i
+			if _G[tex] then
+				_G[tex]:SetTexture()
+			end
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_CreateBarTextures
+DESC: Create empty texture frames for all Titan bars.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanel_CreateBarTextures()
+	-- Create the basic Titan bars (textures)
+	local i, titanTexture
+	local texture_path = TitanPanelGetVar("TexturePath")
+	local bar_name
+	local screenWidth
+	local lastTextureWidth
+	local tex, tex_pre
+
+	-- loop through the bars to set the texture
+	for idx,v in pairs (TitanBarData) do
+		bar_name = TITAN_PANEL_DISPLAY_PREFIX..TitanBarData[idx].name
+		screenWidth = ((_G[bar_name]:GetWidth() or GetScreenWidth()) + 1 ) --/ 2
+		numOfTextures = floor(screenWidth / 256 )
+		numOfTexturesHider = (numOfTextures * 2) + 1
+		lastTextureWidth = screenWidth - (numOfTextures * 256)
+
+		for i = 0, numOfTextures do
+			-- Create textures if they don't exist
+			tex = TITAN_PANEL_BACKGROUND_PREFIX..TitanBarData[idx].name.."_"..i
+			tex_pre = TITAN_PANEL_BACKGROUND_PREFIX..TitanBarData[idx].name.."_"..i-1
+			if not _G[tex] then
+				titanTexture = _G[bar_name]:CreateTexture(tex, "BACKGROUND")
+			else
+				titanTexture = _G[tex]
+			end
+			titanTexture:SetHeight(TITAN_PANEL_BAR_TEXTURE_HEIGHT)
+			if i == numOfTextures then
+				titanTexture:SetWidth(lastTextureWidth)
+			else
+			  titanTexture:SetWidth(256)
+			end
+			titanTexture:ClearAllPoints()
+			if i == 0 then
+				titanTexture:SetPoint("TOPLEFT", bar_name, "TOPLEFT", 0, 0) --  -1, 0)
+			else
+				titanTexture:SetPoint("TOPLEFT", tex_pre, "TOPRIGHT")
+			end
+		end
+		titanTexture:SetWidth(256)
+
+		-- Handle Hider Textures
+		for i = numOfTextures + 1, numOfTexturesHider do
+			tex = TITAN_PANEL_BACKGROUND_PREFIX..TitanBarData[idx].name.."_"..i
+			tex_pre = TITAN_PANEL_BACKGROUND_PREFIX..TitanBarData[idx].name.."_"..i-1
+			if not _G[tex] then
+				titanTexture = _G[bar_name]:CreateTexture(tex, "BACKGROUND")
+			else
+				titanTexture = _G[tex]
+			end
+			titanTexture:SetHeight(TITAN_PANEL_BAR_TEXTURE_HEIGHT)
+			if i == numOfTexturesHider then
+				titanTexture:SetWidth(lastTextureWidth)
+			else
+				titanTexture:SetWidth(256)
+			end
+			titanTexture:ClearAllPoints()
+			titanTexture:SetPoint("TOPLEFT", tex_pre, "TOPRIGHT")
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_SetTexture
+DESC: Set texture frames for the given Titan bar with the user chosen texture (bar graphic).
+VAR: frame - expected to be a Titan bar name (string)
+VAR: position - not used
+OUT:  None
+NOTE:
+- Assumes "TexturePath" contains the user selected texture.
+:NOTE
+--]]
+function TitanPanel_SetTexture(frame, position)
+	if frame and TitanBarData[frame] then
+		-- proceed
+	else
+		return
+	end
+	-- Determine the bar that needs the texture applied.
+	local bar = TitanBarData[frame].name
+	local vert = TitanBarData[frame].vert
+	local tex = "TitanPanelBackground"
+	local tex_pre = tex.."_"..bar.."_"
+
+	-- Create the path & file name to the texture
+	local texture_file = TitanPanelGetVar("TexturePath")..tex..vert.."0"
+	-- include the normal bar (numOfTextures) and hider textures (numOfTexturesHider)
+	for i = 0, numOfTexturesHider do
+		_G[tex_pre..i]:SetTexture(texture_file);
+	end
+end
+
+--------------------------------------------------------------
+--
+-- auto hide event handlers
+--[[ Titan
+NAME: TitanPanelBarButton_OnLeave
+DESC: On leaving the display check if we have to hide the Titan bar. A timer is used - when it expires the bar is hid.
+VAR: self - expected to be a Titan bar
+OUT: None
+--]]
+function TitanPanelBarButton_OnLeave(self)
+	local frame = (self and self:GetName() or nil)
+	local bar = (TitanBarData[frame] and TitanBarData[frame].name or nil)
+
+	-- if auto hide is active then let the timer hide the bar
+	local hide = (bar and TitanPanelGetVar(bar.."_Hide") or nil)
+	if hide then
+		Titan_AutoHide_Timers(frame, "Leave")
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_OnEnter
+DESC: No code - this is a place holder for the XML template.
+VAR: self - expected to be a Titan bar
+OUT: None
+--]]
+function TitanPanelBarButton_OnEnter(self)
+	-- no work to do
+end
+
+--[[ Titan
+NAME: TitanPanelBarButtonHider_OnLeave
+DESC: No code - this is a place holder for the XML template.
+VAR: self - expected to be a Titan bar
+OUT: None
+--]]
+function TitanPanelBarButtonHider_OnLeave(self)
+	-- no work to do
+end
+
+--[[ Titan
+NAME: TitanPanelBarButtonHider_OnEnter
+DESC: On entering the hider check if we need to show the display bar.
+VAR: self - expected to be a Titan hider bar
+OUT: None
+NOTE:
+- No action is taken if the user is on combat.
+:NOTE
+--]]
+function TitanPanelBarButtonHider_OnEnter(self)
+	-- make sure self is valid
+	local index = self and self:GetName() or nil
+	if not index then return end -- sanity check
+
+	-- so the bar does not 'appear' when moused over in combat
+	if TitanPanelGetVar("LockAutoHideInCombat") and InCombatLockdown() then return end
+
+	-- find the relevant bar data
+	local frame = nil
+	for idx,v in pairs (TitanBarData) do
+		if index == TitanBarData[idx].hider then
+			frame = idx
+		end
+	end
+	-- Now process that bar
+	if frame then
+		Titan_AutoHide_Timers(frame, "Enter")
+		TitanPanelBarButton_Show(frame)
+	end
+end
+
+--------------------------------------------------------------
+--
+-- Titan features
+--[[ Titan
+NAME: TitanPanelBarButton_ToggleAlign
+DESC: Align the buttons per the user's new choice.
+VAR: align - left or center
+OUT: None
+--]]
+function TitanPanelBarButton_ToggleAlign(align)
+	-- toggle between left or center
+	if ( TitanPanelGetVar(align) == TITAN_PANEL_BUTTONS_ALIGN_CENTER ) then
+		TitanPanelSetVar(align, TITAN_PANEL_BUTTONS_ALIGN_LEFT);
+	else
+		TitanPanelSetVar(align, TITAN_PANEL_BUTTONS_ALIGN_CENTER);
+	end
+
+	-- Justify button position
+	TitanPanelButton_Justify();
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_ToggleAutoHide
+DESC: Toggle the auto hide of the given Titan bar per the user's new choice.
+VAR: frame - expected to be a Titan bar
+OUT:  None
+--]]
+function TitanPanelBarButton_ToggleAutoHide(frame)
+	local frName = _G[frame]
+	local plugin = (TitanBarData[frame] and TitanBarData[frame].auto_hide_plugin or nil)
+
+	if frName then
+		Titan_AutoHide_ToggleAutoHide(_G[plugin])
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_ToggleScreenAdjust
+DESC: Toggle whether Titan adjusts 'top' frames around Titan bars per the user's new choice.
+VAR:  None
+NOTE:
+- Another addon can tell Titan to NOT adjust some or all frames.
+:NOTE
+--]]
+function TitanPanelBarButton_ToggleScreenAdjust()
+	-- Turn on / off adjusting of other frames around Titan
+	TitanPanelToggleVar("ScreenAdjust");
+	TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_TOP, true)
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_ToggleAuxScreenAdjust
+DESC: Toggle whether Titan adjusts 'bottom' frames around Titan bars per the user's new choice.
+VAR:  None
+OUT:  None
+NOTE:
+- Another addon can tell Titan to NOT adjust some or all frames.
+:NOTE
+--]]
+function TitanPanelBarButton_ToggleAuxScreenAdjust()
+	-- turn on / off adjusting of frames at the bottom of the screen
+	TitanPanelToggleVar("AuxScreenAdjust");
+	TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_BOTTOM, true)
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_ForceLDBLaunchersRight
+DESC: Force all plugins created from LDB addons, visible or not, to be on the right side of the Titan bar. Any visible plugin will be forced to the right side on the same bar it is currently on.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanelBarButton_ForceLDBLaunchersRight()
+	local plugin, index, id;
+	for index, id in pairs(TitanPluginsIndex) do
+		plugin = TitanUtils_GetPlugin(id);
+		if plugin.ldb == "launcher"
+		and not TitanGetVar(id, "DisplayOnRightSide") then
+			TitanToggleVar(id, "DisplayOnRightSide");
+			local button = TitanUtils_GetButton(id);
+			local buttonText = _G[button:GetName()..TITAN_PANEL_TEXT];
+			if not TitanGetVar(id, "ShowIcon") then
+				TitanToggleVar(id, "ShowIcon");
+			end
+			TitanPanelButton_UpdateButton(id);
+			if buttonText then
+				buttonText:SetText("")
+				button:SetWidth(16);
+				TitanPlugins[id].buttonTextFunction = nil;
+				_G["TitanPanel"..id..TITAN_PANEL_BUTTON_TEXT] = nil;
+				if button:IsVisible() then
+					local bar = TitanUtils_GetWhichBar(id)
+					TitanPanel_RemoveButton(id);
+					TitanUtils_AddButtonOnBar(bar, id)
+				end
+			end
+		end
+	end
+end
+
+--[[ local
+NAME: TitanAnchors
+DESC: Helper to create the 'anchor' frames used by other addons that need to adjust so Titan can be visible. The anchor frames are adjusted depending on which Titan bars the user selects to show.
+VAR:  None
+OUT:  None
+NOTE:
+- TitanPanelTopAnchor - the frame at the bottom of the top bar(s) shown.
+- TitanPanelBottomAnchor - the frame at the top of the bottom bar(s) shown.
+:NOTE
+--]]
+local function TitanAnchors()
+	local anchor_top = TitanMovable_GetPanelYOffset(TITAN_PANEL_PLACE_TOP)
+	local anchor_bot = TitanMovable_GetPanelYOffset(TITAN_PANEL_PLACE_BOTTOM)
+	anchor_top = anchor_top <= TITAN_WOW_SCREEN_TOP and anchor_top or TITAN_WOW_SCREEN_TOP
+	anchor_bot = anchor_bot >= TITAN_WOW_SCREEN_BOT and anchor_bot or TITAN_WOW_SCREEN_BOT
+
+	local top_point, top_rel_to, top_rel_point, top_x, top_y = TitanPanelTopAnchor:GetPoint(TitanPanelTopAnchor:GetNumPoints())
+	local bot_point, bot_rel_to, bot_rel_point, bot_x, bot_y = TitanPanelBottomAnchor:GetPoint(TitanPanelBottomAnchor:GetNumPoints())
+	top_y = floor(tonumber(top_y) + 0.5)
+	bot_y = floor(tonumber(bot_y) + 0.5)
+--[[
+TitanDebug("Anc top: "..top_y.." bot: "..bot_y
+.." a_top: "..anchor_top.." a_bot: "..anchor_bot
+)
+--]]
+	if top_y ~= anchor_top then
+		TitanPanelTopAnchor:ClearAllPoints()
+		TitanPanelTopAnchor:SetPoint(top_point, top_rel_to, top_rel_point, top_x, anchor_top);
+--TitanDebug("Anc top: "..top_y.." -> "..anchor_top)
+	end
+	if bot_y ~= anchor_bot then
+		TitanPanelBottomAnchor:ClearAllPoints()
+		TitanPanelBottomAnchor:SetPoint(bot_point, bot_rel_to, bot_rel_point, bot_x, anchor_bot)
+--TitanDebug("Anc bot: "..bot_y.." -> "..anchor_bot)
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_DisplayBarsWanted
+DESC: Show all the Titan bars the user has selected.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanelBarButton_DisplayBarsWanted()
+	-- Check all bars to see if the user has requested they be shown
+	for idx,v in pairs (TitanBarData) do
+		-- Show / hide plus kick auto hide, if needed
+		Titan_AutoHide_Init((_G[TitanBarData[idx].auto_hide_plugin] or nil))
+	end
+
+	-- Set anchors for other addons to use.
+	TitanAnchors()
+
+	-- Adjust other frames because the bars shown / hidden may have changed
+	TitanPanel_AdjustFrames(TITAN_PANEL_PLACE_BOTH, true)
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_HideAllBars
+DESC: This routine will hide all the Titan bars (and hiders) regardless of what the user has selected.
+VAR:  None
+OUT:  None
+NOTE:
+- For example when the pet battle is active. We cannot figure out how to move the pet battle frame so we are punting and hiding Titan...
+- We only need to hide the bars (and hiders) - not adjust frames
+:NOTE
+--]]
+function TitanPanelBarButton_HideAllBars()
+	-- Check all bars to see if the user has requested they be shown
+	for idx,v in pairs (TitanBarData) do
+		local bar = TitanBarData[idx].name
+		local hider = _G[TitanBarData[idx].hider]
+		local hide = TitanBarData[idx].hide
+--[[
+TitanDebug("_HideAllBars: "
+.."idx: "..(idx or "?").." "
+.."bar: "..(bar or "?").." "
+.."hider: "..(TitanBarData[idx].hider or "?").." "
+)
+--]]
+		-- Hide the Titan bar. The hider bar MAY be hidden as well
+		TitanPanelBarButton_Hide(idx)
+		-- In case the user selected auto hide for this bar we need to hide it as well
+		if (TitanPanelGetVar(bar.."_Show")) and (TitanPanelGetVar(bar.."_Hide")) then
+			hider:ClearAllPoints();
+			hider:SetPoint(hide.top.pt, hide.top.rel_fr, hide.top.rel_pt, hide.top.x, hide.top.y);
+			hider:SetPoint(hide.bot.pt, hide.bot.rel_fr, hide.bot.rel_pt, hide.bot.x, hide.bot.y);
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_Show
+DESC: Show / hide the given Titan bar based on the user selection.
+VAR: frame - expected to be a Titan bar name (string)
+OUT:  None
+NOTE:
+- Hide moves rather than just 'not shown'. Otherwise the buttons will stay visible defeating the purpose of hide.
+:NOTE
+--]]
+function TitanPanelBarButton_Show(frame)
+	local display = _G[frame];
+	local bar = (TitanBarData[frame].name or nil)
+	local hide = TitanBarData[frame] and TitanBarData[frame].hide or nil
+	local show = TitanBarData[frame] and TitanBarData[frame].show or nil
+	local hider = _G[TitanBarData[frame].hider] or nil
+
+	if bar and display and hider and show and hide
+	then
+		-- Show the display bar if the user requested it
+		if (TitanPanelGetVar(bar.."_Show")) then
+			if hide and show then
+				display:ClearAllPoints();
+				display:SetPoint(show.top.pt, show.top.rel_fr, show.top.rel_pt, show.top.x, show.top.y);
+				display:SetPoint(show.bot.pt, show.bot.rel_fr, show.bot.rel_pt, show.bot.x, show.bot.y);
+				hider:Hide()
+			end
+		else
+			-- The user has not elected to show this bar
+			display:ClearAllPoints();
+			display:SetPoint(hide.top.pt, hide.top.rel_fr, hide.top.rel_pt, hide.top.x, hide.top.y);
+			display:SetPoint(hide.bot.pt, hide.bot.rel_fr, hide.bot.rel_pt, hide.bot.x, hide.bot.y);
+			hider:ClearAllPoints();
+			hider:SetPoint(hide.top.pt, hide.top.rel_fr, hide.top.rel_pt, hide.top.x, hide.top.y);
+			hider:SetPoint(hide.bot.pt, hide.bot.rel_fr, hide.bot.rel_pt, hide.bot.x, hide.bot.y);
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelBarButton_Hide
+DESC: Hide the given Titan bar based on the user selection.
+VAR: frame - expected to be a Titan bar name (string)
+OUT:  None
+NOTE:
+- Hide moves rather than just 'not shown'. Otherwise the buttons will stay visible defeating the purpose of hide.
+- Also moves the hider bar if auto hide is not selected.
+:NOTE
+--]]
+function TitanPanelBarButton_Hide(frame)
+	if TITAN_PANEL_MOVING == 1 then return end
+
+	local display = _G[frame]
+	local data = TitanBarData[frame]
+	local hider = _G[data.hider]
+	local bar = (data.name or nil)
+	local hide = data.hide or nil
+	local show = data.show or nil
+
+	if display and hider and bar and show and hide then
+		-- This moves rather than hides. If we just hide then
+		-- the plugins will still show.
+		display:ClearAllPoints();
+		display:SetPoint(hide.top.pt, hide.top.rel_fr, hide.top.rel_pt, hide.top.x, hide.top.y);
+		display:SetPoint(hide.bot.pt, hide.bot.rel_fr, hide.bot.rel_pt, hide.bot.x, hide.bot.y);
+		if (TitanPanelGetVar(bar.."_Show")) and (TitanPanelGetVar(bar.."_Hide")) then
+			-- Auto hide is requested so show the hider bar in the right place
+			hider:ClearAllPoints();
+			hider:SetPoint(show.top.pt, show.top.rel_fr, show.top.rel_pt, show.top.x, show.top.y);
+			hider:SetPoint(show.bot.pt, show.bot.rel_fr, show.bot.rel_pt, show.bot.x, show.bot.y);
+			hider:Show()
+		else
+			-- The bar was not requested so also move the hider bar to the right place
+			hider:ClearAllPoints();
+			hider:SetPoint(hide.top.pt, hide.top.rel_fr, hide.top.rel_pt, hide.top.x, hide.top.y);
+			hider:SetPoint(hide.bot.pt, hide.bot.rel_fr, hide.bot.rel_pt, hide.bot.x, hide.bot.y);
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_InitPanelBarButton
+DESC: Set the scale, texture (graphic), and transparancy of all the Titan bars based on the user selection.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanel_InitPanelBarButton()
+	-- Set initial Panel Scale
+	TitanPanel_SetScale();
+	-- Create textures for the first time
+	if numOfTextures == 0 then TitanPanel_CreateBarTextures() end
+
+	-- Reposition textures if needed
+	TitanPanel_CreateBarTextures()
+	for idx,v in pairs (TitanBarData) do
+		TitanPanel_SetTexture(TITAN_PANEL_DISPLAY_PREFIX..TitanBarData[idx].name, TITAN_PANEL_PLACE_TOP);
+	end
+
+	-- Set transparency of the bars
+	local bar = ""
+	local plugin = nil
+	for idx,v in pairs (TitanBarData) do
+		-- Set the transparency of each bar
+		bar = TitanBarData[idx].name
+		_G[idx]:SetAlpha(TitanPanelGetVar(bar.."_Transparency"))
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_InitPanelButtons
+DESC: Show all user selected plugins on the Titan bar(s) then justify per the user selection.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanel_InitPanelButtons()
+	local button
+	local r_prior = {}
+	local l_prior = {}
+	local scale = TitanPanelGetVar("Scale");
+	local button_spacing = TitanPanelGetVar("ButtonSpacing") * scale
+	local icon_spacing = TitanPanelGetVar("IconSpacing") * scale
+
+	local prior = {}
+	-- set prior to the starting offsets
+	-- The right side plugins are set here.
+	-- Justify adjusts the left side start according to the user setting
+	-- The effect is left side plugins has spacing on the right side and
+	-- right side plugins have spacing on the left.
+	for idx,v in pairs (TitanBarData) do
+		local bar = TitanBarData[idx].name
+		local y_off = TitanBarData[idx].plugin_y_offset
+		prior[bar] = {
+			right = {
+				button = TITAN_PANEL_DISPLAY_PREFIX..bar,
+				anchor = "RIGHT",
+				x = 5, -- Offset of first plugin to right side of screen
+				y = y_off,
+				},
+			left = {
+				button = TITAN_PANEL_DISPLAY_PREFIX..bar,
+				anchor = "LEFT",
+				x = 0, -- Justify adjusts - center or not
+				y = y_off,
+				},
+			}
+	end
+--
+	TitanPanelBarButton_DisplayBarsWanted();
+
+	-- Position all the buttons
+	for i = 1, table.maxn(TitanPanelSettings.Buttons) do
+
+		local id = TitanPanelSettings.Buttons[i];
+		if ( TitanUtils_IsPluginRegistered(id) ) then
+			local i = TitanPanel_GetButtonNumber(id);
+			button = TitanUtils_GetButton(id);
+
+			-- If the plugin has asked to be on the right
+			if TitanUtils_ToRight(id) then
+				-- =========================
+				-- position the plugin relative to the prior plugin
+				-- or the bar if it is the 1st
+				r_prior = prior[TitanPanelSettings.Location[i]].right
+				-- =========================
+				button:ClearAllPoints();
+				button:SetPoint("RIGHT", _G[r_prior.button]:GetName(), r_prior.anchor, (-(r_prior.x) * scale), r_prior.y);
+
+				-- =========================
+				-- capture the button for the next plugin
+				r_prior.button = "TitanPanel"..id.."Button"
+				-- set prior[x] the anchor points and offsets for the next plugin
+				r_prior.anchor = "LEFT"
+				r_prior.x = icon_spacing
+				r_prior.y = 0
+				-- =========================
+			else
+				--  handle plugins on the left side of the bar
+				--
+				-- =========================
+				-- position the plugin relative to the prior plugin
+				-- or the bar if it is the 1st
+				l_prior = prior[TitanPanelSettings.Location[i]].left
+				-- =========================
+				--
+				button:ClearAllPoints();
+				button:SetPoint("LEFT", _G[l_prior.button]:GetName(), l_prior.anchor, l_prior.x * scale, l_prior.y);
+
+				-- =========================
+				-- capture the next plugin
+				l_prior.button = "TitanPanel"..id.."Button"
+				-- set prior[x] (anchor points and offsets) for the next plugin
+				l_prior.anchor = "RIGHT"
+				l_prior.x = (button_spacing)
+				l_prior.y = 0
+				-- =========================
+			end
+			button:Show();
+		end
+	end
+	-- Set panel button init flag
+	TITAN_PANEL_BUTTONS_INIT_FLAG = 1;
+	TitanPanelButton_Justify();
+end
+
+--[[ Titan
+NAME: TitanPanel_ReOrder
+DESC: Reorder all the shown all user selected plugins on the Titan bar(s). Typically used after a button has been removed / hidden.
+VAR: index - the index of the plugin removed so the list can be updated
+OUT:  None
+--]]
+function TitanPanel_ReOrder(index)
+	for i = index, table.getn(TitanPanelSettings.Buttons) do
+		TitanPanelSettings.Location[i] = TitanPanelSettings.Location[i+1]
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_RemoveButton
+DESC: Remove a plugin then show all the shown all user selected plugins on the Titan bar(s).
+VAR:  id - the plugin name (string)
+OUT:  None
+NOTE:
+- This cancels all timers of name "TitanPanel"..id as a safeguard to destroy any active plugin timers based on a fixed naming convention : TitanPanel..id, eg. "TitanPanelClock" this prevents "rogue" timers being left behind by lack of an OnHide check
+:NOTE
+--]]
+function TitanPanel_RemoveButton(id)
+	if ( not TitanPanelSettings ) then
+		return;
+	end
+
+	local i = TitanPanel_GetButtonNumber(id)
+	local currentButton = TitanUtils_GetButton(id);
+
+	-- safeguard ...
+	if id then AceTimer.CancelAllTimers("TitanPanel"..id) end
+
+	TitanPanel_ReOrder(i);
+	table.remove(TitanPanelSettings.Buttons, TitanUtils_GetCurrentIndex(TitanPanelSettings.Buttons, id));
+--TitanDebug("_Remove: "..(id or "?").." "..(i or "?"))
+	if currentButton then
+		currentButton:Hide();
+	end
+	-- Show the existing buttons
+	TitanPanel_InitPanelButtons();
+end
+
+--[[ Titan
+NAME: TitanPanel_GetButtonNumber
+DESC: Get the index of the given plugin from the Titan plugin displayed list.
+VAR: id - the plugin name (string)
+OUT: index of the plugin in the Titan plugin list or the end of the list. The routine returns +1 if not found so it is 'safe' to update / add to the Location
+--]]
+function TitanPanel_GetButtonNumber(id)
+	if (TitanPanelSettings) then
+		for i = 1, table.getn(TitanPanelSettings.Buttons) do
+			if(TitanPanelSettings.Buttons[i] == id) then
+				return i;
+			end
+		end
+		return table.getn(TitanPanelSettings.Buttons)+1;
+	else
+		return 0;
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_RefreshPanelButtons
+DESC: Update / refresh each plugin from the Titan plugin list. Used when a Titan option is changed that effects all plugins.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanel_RefreshPanelButtons()
+	if (TitanPanelSettings) then
+		for i = 1, table.getn(TitanPanelSettings.Buttons) do
+			TitanPanelButton_UpdateButton(TitanPanelSettings.Buttons[i], 1);
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelButton_Justify
+DESC: Justify the plugins on each Titan bar. Used when the user changes the 'center' option on a Titan bar.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanelButton_Justify()
+	-- Only the left side buttons are justified.
+	if ( not TITAN_PANEL_BUTTONS_INIT_FLAG or not TitanPanelSettings ) then
+		return;
+	end
+	if InCombatLockdown() then
+--TitanDebug("_Justify during combat!!!")
+		return;
+		-- Issue 856 where some taint is caused if the plugin size is updated during combat. Seems since Mists was released...
+	end
+
+	local bar
+	local vert
+	local y_offset
+	local firstLeftButton
+	local scale = TitanPanelGetVar("Scale");
+	local button_spacing = TitanPanelGetVar("ButtonSpacing") * scale
+	local icon_spacing = TitanPanelGetVar("IconSpacing") * scale
+	local leftWidth = 0;
+	local rightWidth = 0;
+	local counter = 0;
+	local align = 0;
+	local center_offset = 0;
+
+	-- Look at each bar for plugins.
+	for idx,v in pairs (TitanBarData) do
+		bar = TitanBarData[idx].name
+		vert = TitanBarData[idx].vert
+		y_offset = TitanBarData[idx].plugin_y_offset
+		firstLeftButton = TitanUtils_GetButton(TitanPanelSettings.Buttons[TitanUtils_GetFirstButtonOnBar (bar, TITAN_LEFT)])
+		align = TitanPanelGetVar(bar.."_Align")
+		leftWidth = 0;
+		rightWidth = 0;
+		counter = 0;
+		-- If there is a plugin on this bar then justify the first button.
+		-- The other buttons are relative to the first.
+		if ( firstLeftButton ) then
+			if ( align == TITAN_PANEL_BUTTONS_ALIGN_LEFT ) then
+				-- Now offset the plugins
+				firstLeftButton:ClearAllPoints();
+				firstLeftButton:SetPoint("LEFT", idx, "LEFT", 5, y_offset);
+			end
+			-- Center if requested
+			if ( align == TITAN_PANEL_BUTTONS_ALIGN_CENTER ) then
+				leftWidth = 0;
+				rightWidth = 0;
+				counter = 0;
+				-- Calc the total width of the icons so we know where to start
+				for index, id in pairs(TitanPanelSettings.Buttons) do
+					local button = TitanUtils_GetButton(id);
+					if button and button:GetWidth() then
+						if TitanUtils_GetWhichBar(id) == bar then
+							if ( TitanPanelButton_IsIcon(id)
+							or (TitanGetVar(id, "DisplayOnRightSide")) ) then
+								rightWidth = rightWidth
+									+ icon_spacing
+									+ button:GetWidth();
+							else
+								counter = counter + 1;
+								leftWidth = leftWidth
+									+ button_spacing
+									+ button:GetWidth()
+							end
+						end
+					end
+				end
+				-- Now offset the plugins on the bar
+				firstLeftButton:ClearAllPoints();
+				-- remove the last spacing otherwise the buttons appear justified too far left
+				center_offset = (0 - (leftWidth-button_spacing) / 2)
+				firstLeftButton:SetPoint("LEFT", idx, "CENTER", center_offset, y_offset);
+			end
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_SetScale
+DESC: Set the scale of each plugin and each Titan bar.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanel_SetScale()
+	local scale = TitanPanelGetVar("Scale");
+
+	-- Set all the Titan bars
+	for idx,v in pairs (TitanBarData) do
+		local bar_name = TITAN_PANEL_DISPLAY_PREFIX..TitanBarData[idx].name
+		_G[bar_name]:SetScale(scale)
+	end
+	-- Set all the registered plugins
+	for index, value in pairs(TitanPlugins) do
+		if index then
+			TitanUtils_GetButton(index):SetScale(scale);
+		end
+	end
+end
+
+--------------------------------------------------------------
+--
+-- Local routines for Titan menu creation
+--[[ local
+NAME: TitanPanelRightClickMenu_BarOnClick
+DESC: Show / hide a plugin. Used by the Titan (right click) menu.
+VAR: checked - true (show) or false (hide)
+VAR: value - the plugin id
+OUT:  None
+--]]
+local function TitanPanelRightClickMenu_BarOnClick(checked, value)
+	-- we need to know which bar the user clicked to get the menu...
+	local frame = TitanPanel_DropMenu:GetParent():GetName()
+	local bar = TitanBarData[frame].name
+
+	if checked then
+		TitanPanel_RemoveButton(value);
+	else
+		TitanUtils_AddButtonOnBar(bar, value)
+	end
+end
+
+--[[ local
+NAME: TitanPanel_MainMenu
+DESC: Show main Titan (right click) menu.
+VAR:  None
+OUT:  None
+--]]
+local function TitanPanel_MainMenu()
+	local info = {};
+
+	-----------------
+	-- Menu title
+	TitanPanelRightClickMenu_AddTitle(L["TITAN_PANEL_MENU_TITLE"]);
+	TitanPanelRightClickMenu_AddSpacer(L_UIDROPDOWNMENU_MENU_LEVEL);
+
+	TitanPanelRightClickMenu_AddTitle(L["TITAN_PANEL_MENU_PLUGINS"]);
+
+	-----------------
+	-- Plugin Categories
+	for index, id in pairs(L["TITAN_PANEL_MENU_CATEGORIES"]) do
+		info = {};
+		info.notCheckable = true
+		info.text = L["TITAN_PANEL_MENU_CATEGORIES"][index];
+		info.value = "Addons_" .. TITAN_PANEL_BUTTONS_PLUGIN_CATEGORY[index];
+		info.hasArrow = 1;
+		L_UIDropDownMenu_AddButton(info);
+	end
+
+	TitanPanelRightClickMenu_AddSpacer();
+
+	-----------------
+	-- Options - just one button to open the first Titan option screen
+	info = {};
+	info.notCheckable = true
+	info.text = L["TITAN_PANEL_MENU_CONFIGURATION"];
+	info.value = "Bars";
+	info.func = function()
+		InterfaceOptionsFrame_OpenToCategory(L["TITAN_PANEL_MENU_TOP_BARS"])
+	end
+	L_UIDropDownMenu_AddButton(info);
+
+	TitanPanelRightClickMenu_AddSpacer();
+
+	-----------------
+	-- Profiles
+	TitanPanelRightClickMenu_AddTitle(L["TITAN_PANEL_MENU_PROFILES"]);
+
+	-----------------
+	-- Load/Delete
+	info = {};
+	info.notCheckable = true
+	info.text = L["TITAN_PANEL_MENU_MANAGE_SETTINGS"];
+	info.value = "Settings";
+	info.hasArrow = 1;
+	-- lock this menu in combat
+	if InCombatLockdown() then
+		info.disabled = 1;
+		info.hasArrow = nil;
+		info.text = info.text.." "
+			.._G["GREEN_FONT_COLOR_CODE"]
+			..L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"];
+		end
+	L_UIDropDownMenu_AddButton(info);
+
+	-----------------
+	-- Save
+	info = {};
+	info.notCheckable = true
+	info.text = L["TITAN_PANEL_MENU_SAVE_SETTINGS"];
+	info.value = "SettingsCustom";
+	info.func = TitanPanel_SaveCustomProfile;
+	-- lock this menu in combat
+	if InCombatLockdown() then
+		info.disabled = 1;
+		info.text = info.text.." "
+			.._G["GREEN_FONT_COLOR_CODE"]
+			..L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"];
+		end
+	L_UIDropDownMenu_AddButton(info);
+
+	local glob, name, player, server = TitanUtils_GetGlobalProfile()
+	info = {};
+	info.text = "Use Global Profile"
+	info.value = "Use Global Profile"
+	info.func = function()
+		TitanUtils_SetGlobalProfile(not glob, toon)
+		TitanVariables_UseSettings(nil, TITAN_PROFILE_USE)
+	end;
+	info.checked = glob --TitanAllGetVar("GlobalProfileUse")
+	info.keepShownOnClick = nil
+	L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+
+	--local player, server = TitanUtils_ParseName(TitanAllGetVar("GlobalProfileName"))
+	info = {};
+	info.notCheckable = true
+	info.text = "   "..TitanUtils_GetGreenText(server)
+	info.value = "server";
+	L_UIDropDownMenu_AddButton(info);
+
+	info = {};
+	info.notCheckable = true
+	info.text = "      "..TitanUtils_GetGreenText(player)
+	info.value = "player";
+	L_UIDropDownMenu_AddButton(info);
+
+end
+
+--[[ local
+NAME: TitanPanel_ServerSettingsMenu
+DESC: Show list of servers / custom submenu off Profiles/Manage from the Titan (right click) menu.
+VAR:  None
+OUT:  None
+--]]
+local function TitanPanel_ServerSettingsMenu()
+	local info = {};
+	local servers = {};
+	local player = nil;
+	local server = nil;
+	local s, e, ident;
+	local setonce = 0;
+
+	if ( L_UIDROPDOWNMENU_MENU_VALUE == "Settings" ) then
+		TitanPanelRightClickMenu_AddTitle(L["TITAN_PANEL_MENU_PROFILE_SERVERS"],
+			L_UIDROPDOWNMENU_MENU_LEVEL);
+		-- Normal profile per toon
+		for index, id in pairs(TitanSettings.Players) do
+			player, server = TitanUtils_ParseName(index)
+
+			if TitanUtils_GetCurrentIndex(servers, server) == nil then
+				if server ~= TITAN_CUSTOM_PROFILE_POSTFIX then
+					table.insert(servers, server);
+					info = {};
+					info.notCheckable = true
+					info.text = server;
+					info.value = server;
+					info.hasArrow = 1;
+					L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+				end
+			end
+		end
+		-- Custom profiles
+		for index, id in pairs(TitanSettings.Players) do
+			player, server = TitanUtils_ParseName(index)
+
+			if TitanUtils_GetCurrentIndex(servers, server) == nil then
+				if server == TITAN_CUSTOM_PROFILE_POSTFIX then
+					if setonce and setonce == 0 then
+						TitanPanelRightClickMenu_AddTitle("", L_UIDROPDOWNMENU_MENU_LEVEL);
+						TitanPanelRightClickMenu_AddTitle(L["TITAN_PANEL_MENU_PROFILE_CUSTOM"], L_UIDROPDOWNMENU_MENU_LEVEL);
+					setonce = 1;
+					end
+					info = {};
+					info.notCheckable = true
+					info.text = player;
+					info.value = player;
+					info.hasArrow = 1;
+					L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+				end
+			end
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_PlayerSettingsMenu
+DESC: Show list of toons submenu off Profiles/Manage/<server or custom> from the Titan (right click) menu.
+VAR:  None
+OUT:  None
+NOTE:
+- There are 2 level 3 menus possible
+  1) Under profiles, then value could be the server of a saved toon
+  2) Under plugins value could be the options of a plugin
+:NOTE
+--]]
+local function TitanPanel_PlayerSettingsMenu()
+	--
+	local info = {};
+	local player = nil;
+	local server = nil;
+	local s, e, ident;
+	local plugin, profname;
+	local setonce = 0;
+	local off = nil
+
+	--
+	-- Handle the profiles
+	--
+	for index, id in pairs(TitanSettings.Players) do
+		player, server = TitanUtils_ParseName(index)
+		off = (index == TitanSettings.Player)
+			or ((index == TitanAllGetVar("GlobalProfileUse")) and (TitanAllGetVar("GlobalProfileUse")))
+
+		-- handle custom profiles here
+		if server == TITAN_CUSTOM_PROFILE_POSTFIX
+		and player == L_UIDROPDOWNMENU_MENU_VALUE then
+			info = {};
+			info.notCheckable = true
+			info.disabled = TitanAllGetVar("GlobalProfileUse")
+			info.text = L["TITAN_PANEL_MENU_LOAD_SETTINGS"];
+			info.value = index;
+			info.func = function()
+				TitanVariables_UseSettings(index, TITAN_PROFILE_USE)
+			end
+			L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+
+			info = {};
+			info.notCheckable = true
+			info.disabled = off
+			info.text = L["TITAN_PANEL_MENU_DELETE_SETTINGS"];
+			info.value = index;
+			info.func = function()
+				if TitanSettings.Players[info.value] then
+					TitanSettings.Players[info.value] = nil;
+					profname = TitanUtils_ParseName(index)
+					TitanPrint(
+						L["TITAN_PANEL_MENU_PROFILE"]
+						.." '"..profname.."' "
+						..L["TITAN_PANEL_MENU_PROFILE_DELETED"]
+						, "info")
+					TitanPanelRightClickMenu_Close();
+				end
+			end
+			L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+		end -- if server and player
+
+		-- handle regular profiles here
+		if server == L_UIDROPDOWNMENU_MENU_VALUE then
+			-- Set the label once
+			if setonce and setonce == 0 then
+				TitanPanelRightClickMenu_AddTitle(L["TITAN_PANEL_MENU_PROFILE_CHARS"], L_UIDROPDOWNMENU_MENU_LEVEL);
+				setonce = 1;
+			end
+			info = {};
+			info.notCheckable = true
+			info.text = player;
+			info.value = index;
+			info.hasArrow = 1;
+			L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+		end
+	end -- for players
+
+	-- Handle the plugins
+
+	for index, id in pairs(TitanPluginsIndex) do
+		plugin = TitanUtils_GetPlugin(id);
+		if plugin.id and plugin.id == L_UIDROPDOWNMENU_MENU_VALUE then
+			--title
+			info = {};
+			info.text = TitanPlugins[plugin.id].menuText;
+			info.notCheckable = true
+			info.notClickable = 1;
+			info.isTitle = 1;
+			L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+
+			--ShowIcon
+			if plugin.controlVariables.ShowIcon then
+				info = {};
+				info.text = L["TITAN_PANEL_MENU_SHOW_ICON"];
+				info.value = {id, "ShowIcon", nil};
+				info.func = function()
+					TitanPanelRightClickMenu_ToggleVar({id, "ShowIcon", nil})
+				end
+				info.keepShownOnClick = 1;
+				info.checked = TitanGetVar(id, "ShowIcon");
+				info.disabled = nil;
+				L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+			end
+
+			--ShowLabel
+			if plugin.controlVariables.ShowLabelText then
+				info = {};
+				info.text = L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"];
+				info.value = {id, "ShowLabelText", nil};
+				info.func = function()
+					TitanPanelRightClickMenu_ToggleVar({id, "ShowLabelText", nil})
+				end
+				info.keepShownOnClick = 1;
+				info.checked = TitanGetVar(id, "ShowLabelText");
+				info.disabled = nil;
+				L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+			end
+
+			--ShowRegularText (LDB data sources only atm)
+			if plugin.controlVariables.ShowRegularText then
+				info = {};
+				info.text = L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"]
+				info.value = {id, "ShowRegularText", nil};
+				info.func = function()
+					TitanPanelRightClickMenu_ToggleVar({id, "ShowRegularText", nil})
+				end
+				info.keepShownOnClick = 1;
+				info.checked = TitanGetVar(id, "ShowRegularText");
+				info.disabled = nil;
+				L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+			end
+
+			--ShowColoredText
+			if plugin.controlVariables.ShowColoredText then
+				info = {};
+				info.text = L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"];
+				info.value = {id, "ShowColoredText", nil};
+				info.func = function()
+					TitanPanelRightClickMenu_ToggleVar({id, "ShowColoredText", nil})
+				end
+				info.keepShownOnClick = 1;
+				info.checked = TitanGetVar(id, "ShowColoredText");
+				info.disabled = nil;
+				L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+			end
+
+			-- Right-side plugin
+			if plugin.controlVariables.DisplayOnRightSide then
+				info = {};
+				info.text = L["TITAN_PANEL_MENU_LDB_SIDE"];
+				info.func = function ()
+					TitanToggleVar(id, "DisplayOnRightSide")
+					local bar = TitanUtils_GetWhichBar(id)
+					TitanPanel_RemoveButton(id);
+					TitanUtils_AddButtonOnBar(bar, id);
+				end
+				info.checked = TitanGetVar(id, "DisplayOnRightSide");
+				info.disabled = nil;
+				L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+			end
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_SettingsSelectionMenu
+DESC: Show save / load submenu off Profiles/Manage/<server or custom>/<profile> from the Titan (right click) menu.
+VAR:  None
+OUT:  None
+--]]
+local function TitanPanel_SettingsSelectionMenu()
+	local info = {};
+
+	info = {};
+	info.notCheckable = true
+	info.disabled = TitanAllGetVar("GlobalProfileUse")
+	info.text = L["TITAN_PANEL_MENU_LOAD_SETTINGS"];
+	info.value = L_UIDROPDOWNMENU_MENU_VALUE;
+	info.func = function()
+		TitanVariables_UseSettings(L_UIDROPDOWNMENU_MENU_VALUE, TITAN_PROFILE_USE)
+	end
+	L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+
+	info = {};
+	info.notCheckable = true
+	info.disabled = (L_UIDROPDOWNMENU_MENU_VALUE == TitanSettings.Player)
+		or ((L_UIDROPDOWNMENU_MENU_VALUE == TitanAllGetVar("GlobalProfileName"))
+			and (TitanAllGetVar("GlobalProfileUse")))
+	info.text = L["TITAN_PANEL_MENU_DELETE_SETTINGS"];
+	info.value = L_UIDROPDOWNMENU_MENU_VALUE;
+	info.func = function()
+		-- do not delete if current profile - .disabled
+--[[
+		local profilevalue, _, _ = TitanUtils_GetPlayer()
+		if info.value == profilevalue then
+			TitanPrint(L["TITAN_PANEL_ERROR_PROF_DELCURRENT"], "info")
+			TitanPanelRightClickMenu_Close();
+			return;
+		end
+--]]
+		if TitanSettings.Players[info.value] then
+			TitanSettings.Players[info.value] = nil;
+			TitanPrint(
+				L["TITAN_PANEL_MENU_PROFILE"]
+				.." '"..info.value.."' "
+				..L["TITAN_PANEL_MENU_PROFILE_DELETED"]
+				, "info")
+			TitanPanelRightClickMenu_Close();
+		end
+	end
+	L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+end
+
+--[[ Titan
+NAME: TitanPanel_BuildOtherPluginsMenu
+DESC: Show the submenu with list of plugin off the category from the Titan (right click) menu.
+VAR:  None
+OUT:  None
+--]]
+local function TitanPanel_BuildOtherPluginsMenu(frame)
+	local info = {};
+	local plugin;
+
+	for index, id in pairs(TitanPluginsIndex) do
+		plugin = TitanUtils_GetPlugin(id);
+		if not plugin.category then
+			plugin.category = "General";
+		end
+		if ( L_UIDROPDOWNMENU_MENU_VALUE == "Addons_" .. plugin.category ) then
+			if not TitanGetVar(id, "ForceBar")
+				or (TitanGetVar(id, "ForceBar") == TitanBarData[frame].name) then
+				info = {};
+				if plugin.version ~= nil and TitanPanelGetVar("VersionShown") then
+					info.text = plugin.menuText
+						..TitanUtils_GetGreenText(" (v"..plugin.version..")")
+				else
+					info.text = plugin.menuText;
+				end
+				if plugin.controlVariables then
+					info.hasArrow = 1;
+				end
+				info.value = id;
+				info.func = function()
+						local checked = TitanPanel_IsPluginShown(id) or nil;
+						TitanPanelRightClickMenu_BarOnClick(checked, id)
+					end;
+				info.checked = TitanPanel_IsPluginShown(id) or nil
+				info.keepShownOnClick = 1;
+				L_UIDropDownMenu_AddButton(info, L_UIDROPDOWNMENU_MENU_LEVEL);
+			end
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelRightClickMenu_PrepareBarMenu
+DESC: This is the controller to show the proper level of the Titan (right click) menu.
+VAR: self - expected to be the Tian bar that was right clicked
+OUT: None
+--]]
+function TitanPanelRightClickMenu_PrepareBarMenu(self)
+	-- Determine which bar was clicked on
+	-- This MUST match the convention used in TitanPanel.xml to declare
+	-- the dropdown menu. ($parentRightClickMenu)
+	local s, e, frame = string.find(self:GetName(), "(.*)RightClickMenu");
+
+	-- Level 2
+	if ( L_UIDROPDOWNMENU_MENU_LEVEL == 2 ) then
+		TitanPanel_BuildOtherPluginsMenu(frame);
+		TitanPanel_ServerSettingsMenu();
+		return;
+	end
+
+	-- Level 3
+	if ( L_UIDROPDOWNMENU_MENU_LEVEL == 3 ) then
+		TitanPanel_PlayerSettingsMenu();
+		return;
+	end
+
+	-- Level 4
+	if ( L_UIDROPDOWNMENU_MENU_LEVEL == 4 ) then
+		TitanPanel_SettingsSelectionMenu();
+		return;
+	end
+
+	-- Level 1
+	TitanPanel_MainMenu()
+end
+
+--[[ Titan
+NAME: TitanPanel_IsPluginShown
+DESC: Determine if the given plugin is shown on a Titan bar. The Titan bar could be not shown or on auto hide and the plugin will still be 'shown'.
+VAR: id - plugin name (string)
+OUT: int - index of the plugin or nil
+--]]
+function TitanPanel_IsPluginShown(id)
+	if ( id and TitanPanelSettings ) then
+		return TitanUtils_TableContainsValue(TitanPanelSettings.Buttons, id);
+	end
+end
+
+--[[ Titan
+NAME: TitanPanel_GetPluginSide
+DESC: Determine if the given plugin is or would be on right or left of a Titan bar. This returns right or left regardless of whether the plugin is 'shown'.
+VAR: id - plugin name (string)
+OUT: string - "Right" or "Left"
+--]]
+function TitanPanel_GetPluginSide(id)
+	if ( TitanGetVar(id, "DisplayOnRightSide") ) then
+		return TITAN_RIGHT;
+	elseif ( TitanPanelButton_IsIcon(id) ) then
+		return TITAN_RIGHT;
+	else
+		return TITAN_LEFT;
+	end
+end
diff --git a/Titan/TitanPanel.xml b/Titan/TitanPanel.xml
new file mode 100644
index 0000000..f72b77e
--- /dev/null
+++ b/Titan/TitanPanel.xml
@@ -0,0 +1,263 @@
+<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">
+
+	<GameTooltip name="TitanPanelTooltip" inherits="GameTooltipTemplate" parent="UIParent" hidden="true"/>
+	<Button name="TitanPanelBarButtonHiderTemplate" frameStrata="BACKGROUND" toplevel="true" movable="true" parent="UIParent" virtual="true">
+		<Scripts>
+			<OnLoad>
+				self:RegisterForClicks("LeftButtonUp", "RightButtonDown")
+			</OnLoad>
+		</Scripts>
+		<Size>
+			<AbsDimension x="2560" y="24"/>
+		</Size>
+	</Button>
+
+	<!--
+	This is a control frame used to capture events Titan is interested in.
+	Other buttons will be used for display.
+	-->
+	<Button name="TitanPanelBarButton" frameStrata="BACKGROUND" parent="UIParent">
+		<Size>
+			<AbsDimension x="0" y="0"/>
+		</Size>
+	</Button>
+
+	<Button name="Titan_Bar__Display_Template" frameStrata="DIALOG" toplevel="true" movable="true" parent="UIParent" virtual="true">
+		<!-- Used for debug - may delete at some point
+		<ButtonText name="$parent_Text" inherits="GameFontNormalSmall" justifyH="LEFT">
+			<Anchors>
+				<Anchor point="TOPLEFT" relativeTo="$parent" relativePoint="TOPLEFT">
+					<Offset>
+						<AbsDimension x="1" y="-5" />
+					</Offset>
+				</Anchor>
+			</Anchors>
+		</ButtonText>
+		-->
+		<Frames>
+			<Frame name="$parentRightClickMenu" inherits="L_UIDropDownMenuTemplate" hidden="true">
+			</Frame>
+		</Frames>
+	</Button>
+
+	<!--
+	Each Titan bar has 3 components:
+		- Display
+		- Hider
+		- Auto hide plugin
+	-->
+	<Button name="Titan_Bar__Display_Bar" frameStrata="DIALOG" parent="UIParent" inherits="Titan_Bar__Display_Template">
+		<Anchors>
+			<Anchor point="TOPLEFT" relativeTo="UIParent" relativePoint="TOPLEFT">
+				<Offset>
+					<AbsDimension x="0" y="0"/>
+				</Offset>
+			</Anchor>
+			<Anchor point="BOTTOMRIGHT" relativeTo="UIParent" relativePoint="TOPRIGHT">
+				<Offset>
+					<AbsDimension x="0" y="-24"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+	<Button name="Titan_Bar__Hider_Bar" inherits="TitanPanelBarButtonHiderTemplate" hidden="false">
+		<Anchors>
+			<Anchor point="TOPLEFT" relativeTo="UIParent" relativePoint="TOPLEFT">
+				<Offset>
+					<AbsDimension x="0" y="0"/>
+				</Offset>
+			</Anchor>
+			<Anchor point="BOTTOMRIGHT" relativeTo="UIParent" relativePoint="TOPRIGHT">
+				<Offset>
+					<AbsDimension x="0" y="-24"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+
+	<Button name="Titan_Bar__Display_Bar2" frameStrata="DIALOG" parent="UIParent" inherits="Titan_Bar__Display_Template">
+		<Anchors>
+			<Anchor point="TOPLEFT" relativeTo="UIParent" relativePoint="TOPLEFT">
+				<Offset>
+					<AbsDimension x="0" y="-24"/>
+				</Offset>
+			</Anchor>
+			<Anchor point="BOTTOMRIGHT" relativeTo="UIParent" relativePoint="TOPRIGHT">
+				<Offset>
+					<AbsDimension x="0" y="-48"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+	<Button name="Titan_Bar__Hider_Bar2" inherits="TitanPanelBarButtonHiderTemplate" hidden="false">
+		<Anchors>
+			<Anchor point="TOPLEFT" relativeTo="UIParent" relativePoint="TOPLEFT">
+				<Offset>
+					<AbsDimension x="0" y="-24"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+
+	<Button name="Titan_Bar__Display_AuxBar2" frameStrata="DIALOG" parent="UIParent" inherits="Titan_Bar__Display_Template">
+		<Anchors>
+			<Anchor point="BOTTOMLEFT" relativeTo="UIParent" relativePoint="BOTTOMLEFT">
+				<Offset>
+					<AbsDimension x="0" y="24"/>
+				</Offset>
+			</Anchor>
+			<Anchor point="TOPRIGHT" relativeTo="UIParent" relativePoint="BOTTOMRIGHT">
+				<Offset>
+					<AbsDimension x="0" y="48"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+	<Button name="Titan_Bar__Hider_AuxBar2" inherits="TitanPanelBarButtonHiderTemplate" hidden="false">
+		<Anchors>
+			<Anchor point="TOPLEFT" relativeTo="UIParent" relativePoint="TOPLEFT">
+				<Offset>
+					<AbsDimension x="0" y="-24"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+
+	<Button name="Titan_Bar__Display_AuxBar" frameStrata="DIALOG" parent="UIParent" inherits="Titan_Bar__Display_Template">
+		<Anchors>
+			<Anchor point="BOTTOMLEFT" relativeTo="UIParent" relativePoint="BOTTOMLEFT">
+				<Offset>
+					<AbsDimension x="0" y="0"/>
+				</Offset>
+			</Anchor>
+			<Anchor point="TOPRIGHT" relativeTo="UIParent" relativePoint="BOTTOMRIGHT">
+				<Offset>
+					<AbsDimension x="0" y="24"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+	<Button name="Titan_Bar__Hider_AuxBar" inherits="TitanPanelBarButtonHiderTemplate" hidden="false">
+		<Anchors>
+			<Anchor point="TOPLEFT" relativeTo="UIParent" relativePoint="TOPLEFT">
+				<Offset>
+					<AbsDimension x="0" y="-24"/>
+				</Offset>
+			</Anchor>
+		</Anchors>
+	</Button>
+
+	<!--
+	The Auto hide plugin must be wrapped in a frame for the code to work correctly.
+	-->
+	<Frame parent="UIParent">
+		<Frames>
+			<Button name="TitanPanelAutoHide_BarButton" inherits="TitanPanelIconTemplate" frameStrata="FULLSCREEN" toplevel="true">
+				<Scripts>
+					<OnLoad>
+						Titan_AutoHide_OnLoad(self);
+						TitanPanelButton_OnLoad(self);
+					</OnLoad>
+					<OnShow>
+						Titan_AutoHide_OnShow(self);
+					</OnShow>
+					<OnClick>
+						Titan_AutoHide_OnClick(self, button);
+						TitanPanelButton_OnClick(self, button);
+					</OnClick>
+				</Scripts>
+			</Button>
+			<Button name="TitanPanelAutoHide_Bar2Button" inherits="TitanPanelIconTemplate" frameStrata="FULLSCREEN" toplevel="true">
+				<Scripts>
+					<OnLoad>
+						Titan_AutoHide_OnLoad(self);
+						TitanPanelButton_OnLoad(self);
+					</OnLoad>
+					<OnShow>
+						Titan_AutoHide_OnShow(self);
+					</OnShow>
+					<OnClick>
+						Titan_AutoHide_OnClick(self, button);
+						TitanPanelButton_OnClick(self, button);
+					</OnClick>
+				</Scripts>
+			</Button>
+			<Button name="TitanPanelAutoHide_AuxBar2Button" inherits="TitanPanelIconTemplate" frameStrata="FULLSCREEN" toplevel="true">
+				<Scripts>
+					<OnLoad>
+						Titan_AutoHide_OnLoad(self);
+						TitanPanelButton_OnLoad(self);
+					</OnLoad>
+					<OnShow>
+						Titan_AutoHide_OnShow(self);
+					</OnShow>
+					<OnClick>
+						Titan_AutoHide_OnClick(self, button);
+						TitanPanelButton_OnClick(self, button);
+					</OnClick>
+				</Scripts>
+			</Button>
+			<Button name="TitanPanelAutoHide_AuxBarButton" inherits="TitanPanelIconTemplate" frameStrata="FULLSCREEN" toplevel="true">
+		<Scripts>
+			<OnLoad>
+				Titan_AutoHide_OnLoad(self);
+				TitanPanelButton_OnLoad(self);
+			</OnLoad>
+			<OnShow>
+				Titan_AutoHide_OnShow(self);
+			</OnShow>
+			<OnClick>
+				Titan_AutoHide_OnClick(self, button);
+				TitanPanelButton_OnClick(self, button);
+			</OnClick>
+		</Scripts>
+	</Button>
+		</Frames>
+	</Frame>
+	<!--
+	These two frames are placed at the bottom of the top
+	and top of the bottom to give addons a notion of what
+	space Titan is using.
+	A non zero size is needed for the anchors to respond properly.
+	-->
+	<Button name="TitanPanelTopAnchor" frameStrata="DIALOG" parent="UIParent" >
+		<Size>
+			<AbsDimension x="1" y="1" />
+		</Size>
+	</Button>
+	<Button name="TitanPanelBottomAnchor" frameStrata="DIALOG" parent="UIParent" >
+		<Size>
+			<AbsDimension x="1" y="1" />
+		</Size>
+	</Button>
+
+	<!--
+	<Button name="TitanPanelTopTest" frameStrata="DIALOG" parent="UIParent">
+		<Anchors>
+			<Anchor point="TOPLEFT" relativeTo="TitanPanelTopAnchor" relativePoint="BOTTOMLEFT" />
+		</Anchors>
+		<Size>
+			<AbsDimension x="10" y="1" />
+		</Size>
+		<ButtonText name="$parentText" inherits="GameFontNormalSmall" justifyH="LEFT" >
+			<Anchors>
+				<Anchor point="TOPLEFT" relativeTo="$parent" relativePoint="TOPLEFT" />
+			</Anchors>
+		</ButtonText>
+	</Button>
+	<Button name="TitanPanelBottomTest" frameStrata="DIALOG" parent="UIParent">
+		<Anchors>
+			<Anchor point="BOTTOMLEFT" relativeTo="TitanPanelBottomAnchor" relativePoint="TOPLEFT" />
+		</Anchors>
+		<Size>
+			<AbsDimension x="10" y="10" />
+		</Size>
+		<ButtonText name="$parentText" inherits="GameFontNormalSmall" justifyH="LEFT" >
+			<Anchors>
+				<Anchor point="TOPLEFT" relativeTo="$parent" relativePoint="TOPLEFT" />
+			</Anchors>
+		</ButtonText>
+	</Button>
+	-->
+</Ui>
\ No newline at end of file
diff --git a/Titan/TitanPanelChangeLog.txt b/Titan/TitanPanelChangeLog.txt
new file mode 100644
index 0000000..348ed4a
--- /dev/null
+++ b/Titan/TitanPanelChangeLog.txt
@@ -0,0 +1,3 @@
+A complete change history for Titan Panel since version 3.0.0.20000 is available at http://www.titanpanel.org/changes.html.
+
+You may also find the most recent release notes, on our SourceForge page, here : https://sourceforge.net/p/titanpanel/wiki/Titan_Panel_Version_History/
\ No newline at end of file
diff --git a/Titan/TitanPanelSetup-READ_ME_FIRST.txt b/Titan/TitanPanelSetup-READ_ME_FIRST.txt
new file mode 100644
index 0000000..d8409a5
--- /dev/null
+++ b/Titan/TitanPanelSetup-READ_ME_FIRST.txt
@@ -0,0 +1,212 @@
+TitanPanelSetup-READ_ME_FIRST File
+==================================
+
+****************************************************************************************************
+ATTENTION! THE FOLLOWING PLUGINS HAVE BEEN REMOVED OR RENAMED. PLEASE REMOVE THESE FROM YOUR FOLDER!
+- TitanCoords has been renamed TitanLocation to more accurately reflect the information it
+   provides.  REMOVE THE TitanCoords FOLDER FROM YOUR ADDONS FOLDER.
+- TitanGoldTracker has been renamed TitanGold.  REMOVE THE TitanGoldTracker FOLDER FROM YOUR ADDONS
+   FOLDER.
+- TitanAmmo has been removed from the package due to the elimination of ammo/arrows/bolts from the
+   game.  REMOVE THE TitanAmmo FOLDER FROM YOUR ADDONS FOLDER.
+- TitanRegen has been removed from the package due to lack of interest.  Consequently, it may not
+   work with future versions of Titan Panel.  You are free to maintain this plugin if you wish.
+   IF YOU HAVE NO INTEREST IN MAINTAINING IT, REMOVE THE TitanRegen FOLDER FROM YOUR ADDONS FOLDER.
+- BonusScanner has been removed from Titan Panel as it is not part of the core functionality.
+- BrokerBonusScanner has been removed from Titan Panel as it is not part of the core functionality.
+****************************************************************************************************
+
+1. Description
+
+   Titan Panel adds a configurable interface bar/control panel to your WoW UI.
+
+
+2. Installation
+
+   Unzip the contents of the zip file into your ..\Interface\AddOns directory.
+
+   Important Note for Mac Users:  When a zip file contains folders, the auto-unzip function built
+   into Mac OSX will create a new folder to place them in.  If you unzip directly to the AddOns
+   folder, this newly-created folder will prevent Titan from loading.  We recommend Mac users unzip
+   to the desktop, open the Titan folder on the desktop, then select and drag the contents of it to
+   your ../Interface/AddOns folder.  Verify WoW recognizes the addon.  After signing on to WoW,
+   click the AddOns button on the Realm/Character Selection screen.  Titan Panel should be checked
+   by default; if not, check it.  While there, check any unchecked addons, including Titan plugins,
+   you want to display and uncheck any addons you don't want to display.  Now enter the game.
+
+
+3. Setup
+
+   Basic setup includes displaying the desired bars and configuring the plugins.
+
+   - Bars:
+
+      By default, Titan Panel displays the Main Bar (a single bar at the top of the screen).
+      You can, however, display up to 4 independent bars, 1 or 2 at the top of the screen and/or 1
+      or 2 at the bottom.  To change the number and/or location of the bars, right-click a blank
+      area of the Titan Panel and click the Configuration option.  Alternatively, you can click the
+      Game Menu icon on the bottom action bar, select Interface, click the + next to Titan Panel in
+      the Addons tab, and then click Bars.  Use the Bars option to configure the top bars and the
+      Aux Bars option to configure the bottom bars.
+
+         * Show Bar.  Displays the specified bar:  Main Bar is the top bar, Main Bar 2 is the 2nd
+            top bar, Auxiliary Bar is the bottom bar, Auxiliary Bar 2 is the 2nd bottom bar.
+         * Auto-hide.  Causes the bar to appear only when you move the cursor atop the bar;
+            otherwise, the bar is hidden.
+         * Center text.  Changes the location of display (left-side) plugins from left-justified to
+            centered.
+         * Disable screen adjust.  Titan Panel adjusts the Blizzard UI automatically so Titan Panel
+            fits without overlapping frames, such as the minimap.  This allows you to disable this
+            automatic adjustment.  Select this option if you previously selected Auto-hide.
+         * Disable minimap adjust.  Disables the adjustment of the minimap. This is useful in cases
+            you want to enable another addon to specifically handle that frame.
+         * Automatic ticket frame adjust.  Moves the Blizzard Ticket Frame so it's not under the top
+            bar(s).
+         * Automatic log adjust.  Pins the chat log above your highest horizontal action bar, so the
+            chat log doesn't overlap it.  This only works on Blizzard default bars (not custom
+            bars). Because this option has created a lot of confusion, we recommend that you keep
+            this off, manually move the chat log, and lock it.
+         * Automatic bag adjust. Automatically adjusts your bag containers, so they don't overlap
+            the horizontal action bar when you are using bottom bar(s). Unchecking this option is
+            useful in cases you use another addon which specifically handles your container frames.
+
+   - Configuring Plugins using the Interface menu:
+
+      Select the Plugins option and then click the Plugin you want to move or configure.  Not all of
+      the options listed below apply to every plugin.
+
+         * Show plugin.  Check to display the plugin, uncheck to hide.
+         * Show icon.  Check to display the icon, uncheck to hide.
+         * Show label text.  Check to display the name of the plugin, uncheck to hide.
+         * Show colored text.  Check to show the information of display (left-side) plugins in
+            color, uncheck to display the information in white.
+         * Right-side plugin.  Check to move the plugin to the right side.  Applies to Clock and to
+            LDB launchers that aren't coded as launchers.
+         * < Shift Left.  Swaps the plugin with the one to it's immediate left.
+         * > Shift Right.  Swaps the plugin with the one to it's immediate right.
+         * Bar drop-down.  Specifies the bar on which you want the plugin to appear.  Only those
+            bars you have enabled appear in the dropdown.
+
+   - Configuring Plugins using the Titan Panel right-click menu:
+
+         * Right-click a blank area of the Titan Panel bar on which you want to show, hide, or
+            configure plugins.
+         * Hover the cursor over a category of plugins.  Plugins that are displayed on the bar are
+            selected.
+         * Select additional plugins to display or deselect to hide by clicking the plugin's name on
+            the secondary menu.
+         * Select configuration options for individual plugins by clicking the option on the
+            tertiary menu.
+
+   - Configuring Plugins using individual plugin's right-click menu:  Most plugins have a
+      right-click menu for configuring them.
+
+   - Moving plugins using drag-and-drop:
+
+         * To swap plugins, click-and-drag a plugin onto another one.  You can swap between
+            different bars.
+         * To move a plugin to another bar, click-and-drag it to an empty space on the desired bar.
+
+
+4. Customization
+
+   You have numerous ways to customize Titan Panel.  One way is through core functions.  The
+   following core functions appear on the right side of the main bar:
+
+   - AutoHide (Push Pin icon):  Left-clicking this icon causes the bar to only appear when you move
+      the cursor atop the bar.  Otherwise, the bar is hidden.  There is a separate push pin for each
+      bar.
+
+   - Volume Control (speaker icon):  This allows you to override Blizzard's default sound settings.
+
+      * Right-click this icon to display the following Volume Control options:
+         + Show Sound/Voice options.  Displays the Sound & Voice window.  Here you can control
+            volume using Blizzard's interface.
+         + Override Blizzard Volume Settings.  Select this to control volume using Titan Panel's
+            interface.
+         + Hide.  Removes the plugin from the bar.
+      * Left-click this icon to display 6 slider bars with which you can control specific volume
+         levels.  You MUST select Override Blizzard Volume Settings from the right-click options
+         menu if you want to control the volume with these sliders.
+
+   - Clock:
+
+      * In addition to the options in the Interface menu, right-clicking the Clock gives access to
+         the following options: Show Local Time, Show Server Time, Show Server Adjusted Time, Hide
+         Time button, and Hide Calendar Button.
+      * Left-clicking allows you to adjust the server time and change the time to 24-hour format.
+
+   The following core functions appear in Blizzard's Interface Addons frame:
+
+   - Tooltips and Frames:
+
+      * Hide tooltips in combat.  Turns off tooltips in combat so you're not distracted while
+         fighting.
+      * Show tooltips.  Displays a tooltip when you hover the cursor over a button/icon.
+      * Lock buttons.  Locks the buttons in place, keeping you from inadvertently moving them.
+      * Show plugin versions.  Displays the version number when a plugin is selected from the
+         Interface menu.
+      * Force LDB launchers to right-side.  Use when LDB launchers are not coded as such.
+      * Refresh plugins.  Use when a plugin does not update.  If you need to use this, and it fixes
+         the display, contact the developer.
+      * Reset Titan Panel to Default.  For emergency use only.
+
+   - Scale and Font:
+
+      * UI Scale.  Controls the scale of the User Interface from 64% to 100%.
+      * Titan Panel Scale.  Controls the size of Titan Panel from 75% to 125%.
+      * Button Spacing.  Controls the separation between left-side plugins from 5 pixels to 80
+         pixels.
+      * Tooltip Font Scale.  Controls the size of the plugins' Tooltip Fonts from 50% to 130%.
+      * Disable Tooltip Font Scale.  Overrides the setting on the Tooltip Font Scale slider.
+      * Panel Font drop-down.  Changes the font type used.
+      * Font Size.  Changes the size of the specified font.
+      * Titan Panel Frame Strata drop-down.  Changes the depth you want the bars to be relative to
+         other frames.
+
+   - Transparency:  Controls the transparency of each of the bars and the tooltips.
+
+
+   All the built-ins, and many third-party plugins, give you the option to disable them individually
+   by right-clicking the icon and selecting Hide.  Another way to customize Titan Panel is by
+   changing skins.  Titan comes with a large selection of skins.  You control skins from the
+   Interface Addon menu.
+
+   - Skins:  Here you can select a skin from the Skin List drop-down or Reset to Defaults.
+
+   - Skins-Custom:  Use this screen to add a custom skin to the Skin List or remove one.
+
+
+5. Profiles
+
+   Profiles allow you to have different configurations for individual characters.  The
+   first time you configure Titan Panel, it automatically saves the settings in a profile, using the
+   character's name.  It will continue to use that profile for other characters until you change
+   your configuration and save it.  There are two ways to manage profiles:
+
+   - Titan Panel Right-Click Menu:
+
+      * Profiles Manage.  Allows you to replace the current character's profile with another another
+         character's profile.  Also allows you to delete a profile.
+      * Profiles Save.  Allows you to save the current character's settings under a user-specified
+         name.
+
+   - Interface Addons Menu:  Gives you the same options to manage and save profiles as the
+      right-click menu does.
+
+
+6. Bug Reporting
+
+   If you encounter a bug, open an Issue at our SourceForge site:
+
+      https://sourceforge.net/p/titanpanel/tickets/?source=navbar
+
+   Please include as many details as possible, including a complete list of addons.  It is helpful
+   to use a bug capturing addon, such as !Swatter (part of Gatherer addon); copy and paste the
+   entire text into the Issue screen.  We will post our troubleshooting results and recommendations
+   there, so check back often.  Before opening the Issue, please Search All Issues to see if the bug
+   has already been reported and fixed.  Also, check the download sites for information on the
+   download page, such as:
+
+      http://www.wowinterface.com/downloads/fileinfo.php?id=8092
+      http://www.curse.com/addons/wow/titan-panel
\ No newline at end of file
diff --git a/Titan/TitanPanelTemplate.lua b/Titan/TitanPanelTemplate.lua
new file mode 100644
index 0000000..9dd6e1f
--- /dev/null
+++ b/Titan/TitanPanelTemplate.lua
@@ -0,0 +1,1048 @@
+--[[ File
+NAME: TitanPanelTemplate.lua
+DESC: Contains the routines to handle a frame created as a Titan plugin.
+--]]
+--[[ API
+NAME: TitanPanelTemplate overview
+DESC: See TitanPanelButtonTemplate.xml also.
+
+A Titan plugin is a frame created using one of the button types in TitanPanelButtonTemplate.xml which inherits TitanPanelButtonTemplate.
+The available plugin types are:
+TitanPanelTextTemplate - A frame that only displays text ("$parentText")
+TitanPanelIconTemplate - A frame that only displays an icon ("$parentIcon")
+TitanPanelComboTemplate - A frame that displays an icon then text ("$parentIcon"  "$parentText")
+
+Most plugins use the combo template.
+
+TitanPanelButtonTemplate.xml contains other templates available to be used.
+TitanOptionsSliderTemplate - A frame that contains the basics of a slider control. See TitanVolume for an example.
+TitanPanelChildButtonTemplate - A frame that allows a plugin within a plugin. The older version of TitanGold was an example. This may not be used anymore.
+
+Each template contains:
+- a frame to handle a menu invoked by a right mouse click ("$parentRightClickMenu")
+- default event handlers for
+			<OnLoad>
+				TitanPanelButton_OnLoad(self);
+			</OnLoad>
+			<OnShow>
+				TitanPanelButton_OnShow(self);
+			</OnShow>
+			<OnClick>
+				TitanPanelButton_OnClick(self, button);
+			</OnClick>
+			<OnEnter>
+				TitanPanelButton_OnEnter(self);
+			</OnEnter>
+			<OnLeave>
+				TitanPanelButton_OnLeave(self);
+			</OnLeave>
+If these events are overridden then the default routine needs to be included!
+:DESC
+--]]
+
+-- Globals
+
+-- Constants
+local TITAN_PANEL_LABEL_SEPARATOR = "  "
+local TITAN_PANEL_BUTTON_WIDTH_CHANGE_TOLERANCE = 10;
+local TITAN_PANEL_BUTTON_TYPE_TEXT = 1;
+local TITAN_PANEL_BUTTON_TYPE_ICON = 2;
+local TITAN_PANEL_BUTTON_TYPE_COMBO = 3;
+local TITAN_PANEL_BUTTON_TYPE_CUSTOM = 4;
+local pluginOnEnter = nil;
+local TITAN_PANEL_MOVE_ADDON = nil;
+local TITAN_PANEL_DROPOFF_ADDON = nil;
+
+-- Library instances
+local LibQTip = nil
+local _G = getfenv(0);
+local InCombatLockdown	= _G.InCombatLockdown;
+local media = LibStub("LibSharedMedia-3.0")
+
+--[[ local
+NAME: TitanTooltip_AddTooltipText
+DESC: Helper to add a line of tooltip text to the tooltip.
+VAR:  text - string
+OUT:  None
+NOTE:
+- Append a "\n" to the end if there is not one already there
+:NOTE
+--]]
+local function TitanTooltip_AddTooltipText(text)
+	if ( text ) then
+		-- Append a "\n" to the end
+		if ( string.sub(text, -1, -1) ~= "\n" ) then
+			text = text.."\n";
+		end
+
+		-- See if the string is intended for a double column
+		for text1, text2 in string.gmatch(text, "([^\t\n]*)\t?([^\t\n]*)\n") do
+			if ( text2 ~= "" ) then
+				-- Add as double wide
+				GameTooltip:AddDoubleLine(text1, text2);
+			elseif ( text1 ~= "" ) then
+				-- Add single column line
+				GameTooltip:AddLine(text1);
+			else
+				-- Assume a blank line
+				GameTooltip:AddLine("\n");
+			end
+		end
+	end
+end
+
+--[[ local
+NAME: TitanTooltip_SetOwnerPosition
+DESC: Set both the parent and the position of GameTooltip for the plugin tooltip.
+VAR: parent - reference to the frame to attach the tooltip to
+VAR: anchorPoint - tooltip anchor location (side or corner) to use
+VAR: relativeToFrame - string name name of the frame, usually the plugin), to attach the tooltip to
+VAR: relativePoint - parent anchor location (side or corner) to use
+VAR: xOffset - X offset from the anchor point
+VAR: yOffset - Y offset from the anchor point
+VAR: frame - reference to the tooltip
+OUT:  None
+--]]
+local function TitanTooltip_SetOwnerPosition(parent, anchorPoint, relativeToFrame, relativePoint, xOffset, yOffset, frame)
+	if not frame then
+		frame = _G["GameTooltip"]
+	end
+	frame:SetOwner(parent, "ANCHOR_NONE");
+	frame:SetPoint(anchorPoint, relativeToFrame, relativePoint,
+		xOffset, yOffset);
+	-- set alpha (transparency) for the Game Tooltip
+	local red, green, blue = frame:GetBackdropColor();
+	local red2, green2, blue2 = frame:GetBackdropBorderColor();
+	local tool_trans = TitanPanelGetVar("TooltipTrans")
+	frame:SetBackdropColor(red,green,blue,tool_trans);
+	frame:SetBackdropBorderColor(red2,green2,blue2,tool_trans);
+	-- set font size for the Game Tooltip
+	if not TitanPanelGetVar("DisableTooltipFont") then
+		if TitanTooltipScaleSet < 1 then
+		TitanTooltipOrigScale = frame:GetScale();
+		TitanTooltipScaleSet = TitanTooltipScaleSet + 1;
+		end
+		frame:SetScale(TitanPanelGetVar("TooltipFont"));
+	end
+end
+
+--[[ local
+NAME: TitanTooltip_SetGameTooltip
+DESC: Helper to set the tooltip of the given Titan plugin.
+First check for a custom function. If no function then use the plugin tooltip title and text.
+VAR: self - frame reference of the plugin
+OUT:  None
+NOTE:
+- If a custom function is given pcall (protected call) is used in case the function errors out. Currently the error is allowed to occur silently because it could generate a lot of text to chat.
+:NOTE
+--]]
+local function TitanTooltip_SetGameTooltip(self)
+	if ( self.tooltipCustomFunction ) then
+--[
+		local tmp_txt = ""
+		local call_success
+		call_success, -- for pcall
+			tmp_txt = pcall (self.tooltipCustomFunction)
+--]]
+--		self.tooltipCustomFunction();
+	elseif ( self.tooltipTitle ) then
+		GameTooltip:SetText(self.tooltipTitle,
+			HIGHLIGHT_FONT_COLOR.r, HIGHLIGHT_FONT_COLOR.g, HIGHLIGHT_FONT_COLOR.b);
+		if ( self.tooltipText ) then
+			TitanTooltip_AddTooltipText(self.tooltipText);
+		end
+	end
+
+	GameTooltip:Show();
+end
+
+--[[ local
+NAME: TitanTooltip_SetPanelTooltip
+DESC: Helper to set the screen position of the tooltip of the given Titan plugin.
+VAR: self - frame reference of the plugin
+VAR: id - string name of the plugin
+VAR: frame - reference to the tooltip
+OUT:  None
+--]]
+local function TitanTooltip_SetPanelTooltip(self, id, frame)
+	-- sanity checks
+	if not TitanPanelGetVar("ToolTipsShown")
+	or (TitanPanelGetVar("HideTipsInCombat") and InCombatLockdown()) then return end
+
+	if not self.tooltipCustomFunction and not self.tooltipTitle then return end
+
+	-- Set GameTooltip
+	local button = TitanUtils_GetButton(id);
+	local scale = TitanPanelGetVar("Scale");
+	local offscreenX, offscreenY;
+	local i = TitanPanel_GetButtonNumber(id);
+	local bar = TITAN_PANEL_DISPLAY_PREFIX..TitanUtils_GetWhichBar(id)
+	local vert = TitanBarData[bar].vert
+	-- Get TOP or BOTTOM for the anchor and relative anchor
+	local rel_pt, pt
+	if vert == TITAN_TOP then
+		pt = "TOP"
+		rel_pt = "BOTTOM"
+	else
+		pt = "BOTTOM"
+		rel_pt = "TOP"
+	end
+
+	TitanTooltip_SetOwnerPosition(button, pt.."LEFT",
+		button:GetName(), rel_pt.."LEFT", -10, 0, frame) --4 * scale);
+	TitanTooltip_SetGameTooltip(self);
+
+	-- Adjust GameTooltip position if it's off the screen
+	offscreenX, offscreenY = TitanUtils_GetOffscreen(GameTooltip);
+	if ( offscreenX == -1 ) then
+		TitanTooltip_SetOwnerPosition(button, pt.."LEFT", bar,
+			rel_pt.."LEFT", 0, 0, frame)
+		TitanTooltip_SetGameTooltip(self);
+	elseif ( offscreenX == 1 ) then
+		TitanTooltip_SetOwnerPosition(button, pt.."RIGHT", bar,
+			rel_pt.."RIGHT", 0, 0, frame)
+		TitanTooltip_SetGameTooltip(self);
+	end
+end
+
+--[[ local
+NAME: TitanPanelButton_SetTooltip
+DESC: Set the tooltip of the given Titan plugin.
+VAR: self - frame reference of the plugin
+VAR: id - string name of the plugin
+--]]
+local function TitanPanelButton_SetTooltip(self, id)
+	-- ensure that the 'self' passed is a valid frame reference
+	if not self:GetName() then return end
+
+	self.tooltipCustomFunction = nil;
+	if (id and TitanUtils_IsPluginRegistered(id)) then
+		local plugin = TitanUtils_GetPlugin(id);
+		if ( plugin.tooltipCustomFunction ) then
+			self.tooltipCustomFunction = plugin.tooltipCustomFunction;
+			TitanTooltip_SetPanelTooltip(self, id);
+		elseif ( plugin.tooltipTitle ) then
+			self.tooltipTitle = plugin.tooltipTitle;
+			local tooltipTextFunc = _G[plugin.tooltipTextFunction];
+			if ( tooltipTextFunc ) then
+				local tmp_txt = ""
+				local call_success, -- for pcall
+					tmp_txt = pcall (tooltipTextFunc);
+				self.tooltipText = tmp_txt
+--				self.tooltipText = tooltipTextFunc();
+			end
+			TitanTooltip_SetPanelTooltip(self, id);
+		end
+	end
+end
+
+--[[ local
+NAME: TitanPanelButton_IsText
+DESC: Is the given Titan plugin of type text?
+VAR: id - string name of the plugin
+OUT: boolean
+--]]
+local function TitanPanelButton_IsText(id)
+	if (TitanPanelButton_GetType(id) == TITAN_PANEL_BUTTON_TYPE_TEXT) then
+		return 1;
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelButton_IsIcon
+DESC: Is the given Titan plugin of type icon?
+VAR: id - string name of the plugin
+OUT: boolean
+--]]
+function TitanPanelButton_IsIcon(id)
+	if (TitanPanelButton_GetType(id) == TITAN_PANEL_BUTTON_TYPE_ICON) then
+		return 1;
+	end
+end
+
+--[[ local
+NAME: TitanPanelButton_IsCombo
+DESC: Is the given Titan plugin of type combo?
+VAR:  id - string name of the plugin
+OUT: boolean
+--]]
+local function TitanPanelButton_IsCombo(id)
+	if (TitanPanelButton_GetType(id) == TITAN_PANEL_BUTTON_TYPE_COMBO) then
+		return 1;
+	end
+end
+
+--[[ local
+NAME: TitanPanelButton_IsCustom
+DESC: Is the given Titan plugin of type custom?
+VAR: id - string name of the plugin
+OUT: boolean
+--]]
+local function TitanPanelButton_IsCustom(id)
+	if (TitanPanelButton_GetType(id) == TITAN_PANEL_BUTTON_TYPE_CUSTOM) then
+		return 1;
+	end
+end
+
+--[[ local
+NAME: TitanPanelButton_OnDragStart
+DESC: Handle the OnDragStart event of the given Titan plugin.
+VAR:  self - frame reference of the plugin
+VAR: ChildButton - boolean
+OUT:  None
+NOTE:
+- Do nothing if the user has locked plugins or if in combat.
+- Set the .isMoving of the plugin (frame) so other routine can check it.
+- Set TITAN_PANEL_MOVING so any Titan routine will know a 'drag & drop' is in progress.
+- Set TITAN_PANEL_MOVE_ADDON so sanity checks can be done on the 'drop'.
+:NOTE
+--]]
+local function TitanPanelButton_OnDragStart(self, ChildButton)
+	if TitanPanelGetVar("LockButtons") or InCombatLockdown() then return end
+
+	local frname = self;
+	if ChildButton then
+		frname = self:GetParent();
+	end
+
+	-- Clear button positions or we'll grab the button and all buttons 'after'
+	local i,j;
+	for i, j in pairs(TitanPanelSettings.Buttons) do
+		local pluginid = _G["TitanPanel"..TitanPanelSettings.Buttons[i].."Button"];
+		if pluginid then
+			pluginid:ClearAllPoints()
+		end
+	end
+
+	-- Start the drag; close any tooltips and open control frames
+	frname:StartMoving();
+	frname.isMoving = true;
+	TitanUtils_CloseAllControlFrames();
+	TitanPanelRightClickMenu_Close();
+	if AceLibrary then
+		if AceLibrary:HasInstance("Dewdrop-2.0") then
+			AceLibrary("Dewdrop-2.0"):Close()
+		end
+		if AceLibrary:HasInstance("Tablet-2.0") then
+			AceLibrary("Tablet-2.0"):Close()
+		end
+	end
+	GameTooltip:Hide();
+	-- LibQTip-1.0 support code
+	LibQTip = LibStub("LibQTip-1.0", true)
+	if LibQTip then
+		local key, tip
+		for key, tip in LibQTip:IterateTooltips() do
+			if tip then
+				local _, relativeTo = tip:GetPoint()
+					if relativeTo and relativeTo:GetName() == self:GetName() then
+						tip:Hide()
+						break
+					end
+			end
+		end
+	end
+	-- /LibQTip-1.0 support code
+
+	-- Hold the plugin id so we can do checks on the drop
+	TITAN_PANEL_MOVE_ADDON = TitanUtils_GetButtonID(self:GetName());
+	if ChildButton then
+		TITAN_PANEL_MOVE_ADDON =
+			TitanUtils_GetButtonID(self:GetParent():GetName());
+	end
+	-- Tell Titan that a drag & drop is in process
+	TITAN_PANEL_MOVING = 1;
+	-- Store the OnEnter handler so the tooltip does not show - or other oddities
+	pluginOnEnter = self:GetScript("OnEnter")
+	self:SetScript("OnEnter", nil)
+end
+
+--[[ local
+NAME: TitanPanelButton_OnDragStop
+DESC: Handle the OnDragStop event of the given Titan plugin.
+VAR: self - frame reference of the plugin
+VAR: ChildButton - boolean
+OUT:  None
+NOTE:
+- Clear the .isMoving of the plugin (frame).
+- Clear TITAN_PANEL_MOVING.
+- Clear TITAN_PANEL_MOVE_ADDON.
+:NOTE
+--]]
+local function TitanPanelButton_OnDragStop(self, ChildButton)
+	if TitanPanelGetVar("LockButtons") then
+		return
+	end
+	local ok_to_move = true
+	local nonmovableFrom = false;
+	local nonmovableTo = false;
+	local frname = self;
+	if ChildButton then
+		frname = self:GetParent();
+	end
+	if TITAN_PANEL_MOVING == 1 then
+		frname:StopMovingOrSizing();
+		frname.isMoving = false;
+		TITAN_PANEL_MOVING = 0;
+
+		-- See if the plugin is supposed to stay on the bar it is on
+		if TitanGetVar(TITAN_PANEL_MOVE_ADDON, "ForceBar") then
+			ok_to_move = false
+		end
+
+		-- eventually there could be several reasons to not allow
+		-- the plugin to move
+		if ok_to_move then
+			local i,j;
+			for i, j in pairs(TitanPanelSettings.Buttons) do
+				local pluginid =
+					_G["TitanPanel"..TitanPanelSettings.Buttons[i].."Button"];
+				if (pluginid and MouseIsOver(pluginid)) and frname ~= pluginid then
+					TITAN_PANEL_DROPOFF_ADDON = TitanPanelSettings.Buttons[i];
+				end
+			end
+
+			-- switching sides is not allowed
+			nonmovableFrom = TitanUtils_ToRight(TITAN_PANEL_MOVE_ADDON)
+			nonmovableTo = TitanUtils_ToRight(TITAN_PANEL_DROPOFF_ADDON)
+			if nonmovableTo ~= nonmovableFrom then
+				TITAN_PANEL_DROPOFF_ADDON = nil;
+			end
+
+			if TITAN_PANEL_DROPOFF_ADDON == nil then
+				-- See if the plugin was dropped on a bar rather than
+				-- another plugin.
+				local bar
+				local tbar = nil
+				-- Find which bar it was dropped on
+				for idx,v in pairs(TitanBarData) do
+					bar = idx
+					if (bar and MouseIsOver(_G[bar])) then
+						tbar = bar
+					end
+				end
+
+				if tbar then
+					TitanPanel_RemoveButton(TITAN_PANEL_MOVE_ADDON)
+					TitanUtils_AddButtonOnBar(TitanBarData[tbar].name, TITAN_PANEL_MOVE_ADDON)
+				else
+					-- not sure what the user did...
+				end
+			else
+				-- plugin was dropped on another plugin - swap (for now)
+				local dropoff = TitanUtils_GetCurrentIndex(TitanPanelSettings.Buttons
+					,TITAN_PANEL_DROPOFF_ADDON);
+				local pickup = TitanUtils_GetCurrentIndex(TitanPanelSettings.Buttons
+					,TITAN_PANEL_MOVE_ADDON);
+				local dropoffbar = TitanUtils_GetWhichBar(TITAN_PANEL_DROPOFF_ADDON);
+				local pickupbar = TitanUtils_GetWhichBar(TITAN_PANEL_MOVE_ADDON);
+
+				if dropoff ~= nil and dropoff ~= "" then
+		-- TODO: Change to 'insert' rather than swap
+					TitanPanelSettings.Buttons[dropoff] = TITAN_PANEL_MOVE_ADDON;
+					TitanPanelSettings.Location[dropoff] = dropoffbar;
+					TitanPanelSettings.Buttons[pickup] = TITAN_PANEL_DROPOFF_ADDON;
+					TitanPanelSettings.Location[pickup] = pickupbar;
+				end
+			end
+		end
+
+		-- This is important! The start drag cleared the button positions so
+		-- the buttons need to be put back properly.
+		TitanPanel_InitPanelButtons();
+		TITAN_PANEL_MOVE_ADDON = nil;
+		TITAN_PANEL_DROPOFF_ADDON = nil;
+		-- Restore the OnEnter script handler
+		if pluginOnEnter then self:SetScript("OnEnter", pluginOnEnter) end
+		pluginOnEnter = nil;
+	end
+end
+
+--[[ local
+NAME: TitanTooltip_SetOwnerPosition
+DESC: Set both the parent and the position of GameTooltip for the plugin tooltip.
+VAR: parent - reference to the frame to attach the tooltip to
+VAR: anchorPoint - tooltip anchor location (side or corner) to use
+VAR: relativeToFrame - string name name of the frame, usually the plugin), to attach the tooltip to
+VAR: relativePoint - parent anchor location (side or corner) to use
+VAR: xOffset - X offset from the anchor point
+VAR: yOffset - Y offset from the anchor point
+VAR: frame - reference to the tooltip
+OUT:  None
+--]]
+local function TitanTooltip_SetOwnerPosition(parent, anchorPoint, relativeToFrame, relativePoint, xOffset, yOffset, frame)
+	if not frame then
+		frame = _G["GameTooltip"]
+	end
+	frame:SetOwner(parent, "ANCHOR_NONE");
+	frame:SetPoint(anchorPoint, relativeToFrame, relativePoint,
+		xOffset, yOffset);
+	-- set alpha (transparency) for the Game Tooltip
+	local red, green, blue = frame:GetBackdropColor();
+	local red2, green2, blue2 = frame:GetBackdropBorderColor();
+	local tool_trans = TitanPanelGetVar("TooltipTrans")
+	frame:SetBackdropColor(red,green,blue,tool_trans);
+	frame:SetBackdropBorderColor(red2,green2,blue2,tool_trans);
+	-- set font size for the Game Tooltip
+	if not TitanPanelGetVar("DisableTooltipFont") then
+		if TitanTooltipScaleSet < 1 then
+		TitanTooltipOrigScale = frame:GetScale();
+		TitanTooltipScaleSet = TitanTooltipScaleSet + 1;
+		end
+		frame:SetScale(TitanPanelGetVar("TooltipFont"));
+	end
+end
+
+--[[ API
+NAME: TitanOptionSlider_TooltipText
+DESC: Set the color of the tooltip text to normal (white) with the value in green.
+VAR: text - the label for value
+VAR: value - the value
+OUT: string - encoded color string of text and value
+--]]
+function TitanOptionSlider_TooltipText(text, value)
+	return text .. GREEN_FONT_COLOR_CODE .. value .. FONT_COLOR_CODE_CLOSE;
+end
+
+--[[ API
+NAME: TitanPanelButton_OnLoad
+DESC: Handle the OnLoad event of the requested Titan plugin. Ensure the plugin is set to be registered.
+VAR: isChildButton - boolean
+NOTE:
+- This is called from the Titan plugin frame in the OnLoad event - usually as the frame is created.
+- This starts the plugin registration process. See TitanUtils for more details on plugin registration.
+- The plugin registration is a two step process because not all addons create Titan plugins in the frame create. The Titan feature of converting LDB addons to Titan plugins is an example.
+:NOTE
+--]]
+function TitanPanelButton_OnLoad(self, isChildButton) -- Used by plugins
+	TitanUtils_PluginToRegister(self, isChildButton)
+end
+
+--[[ API
+NAME: TitanPanelPluginHandle_OnUpdate
+DESC: A method to refresh the display of a Titan plugin.
+VAR: table - the frame of the plugin
+VAR: oldarg - nil or command
+NOTE:
+- This is used by some plugins. It is not used within Titan.
+- The expected usage is either:
+1) Table contains {<plugin id>, <update command>}
+2) table = <plugin id> and oldarg = <update command>
+- oldarg - nil or command
+1 = refresh button
+2 = refresh tooltip
+3 = refresh button and tooltip
+:NOTE
+--]]
+function TitanPanelPluginHandle_OnUpdate(table, oldarg) -- Used by plugins
+	local id, updateType = nil, nil
+	-- set the id and updateType
+	-- old method
+	if table and type(table) == "string" and oldarg then
+		id = table
+		updateType = oldarg
+	end
+	-- new method
+	if table and type(table) == "table" then
+		if table[1] then id = table[1] end
+		if table[2] then updateType = table[2] end
+	end
+
+	-- id is required
+	if id then
+		if updateType == TITAN_PANEL_UPDATE_BUTTON
+		or updateType == TITAN_PANEL_UPDATE_ALL then
+			TitanPanelButton_UpdateButton(id)
+		end
+
+		if (updateType == TITAN_PANEL_UPDATE_TOOLTIP
+		or updateType == TITAN_PANEL_UPDATE_ALL)
+		and MouseIsOver(_G["TitanPanel"..id.."Button"]) then
+			if TitanPanelRightClickMenu_IsVisible() or TITAN_PANEL_MOVING == 1 then
+				return
+			end
+			TitanPanelButton_SetTooltip(_G["TitanPanel"..id.."Button"], id)
+		end
+	end
+end
+
+--[[ API
+NAME: TitanPanelDetectPluginMethod
+DESC: Poorly named routine that sets the OnDragStart & OnDragStop scripts of a Titan plugin.
+VAR: id - the string name of the plugin
+VAR: isChildButton - boolean
+--]]
+function TitanPanelDetectPluginMethod(id, isChildButton)
+	-- Ensure the id is not nil
+	if not id then return end
+	local TitanPluginframe = _G["TitanPanel"..id.."Button"];
+	if isChildButton then
+		TitanPluginframe = _G[id];
+	end
+	-- Ensure the frame is valid
+	if not TitanPluginframe and TitanPluginframe:GetName() then return end -- sanity check...
+
+	-- Set the OnDragStart script
+	TitanPluginframe:SetScript("OnDragStart", function(self)
+		if not IsShiftKeyDown()
+		and not IsControlKeyDown()
+		and not IsAltKeyDown() then
+			if isChildButton then
+				TitanPanelButton_OnDragStart(self, true);
+			else
+				TitanPanelButton_OnDragStart(self);
+			end
+		end
+	end)
+
+	-- Set the OnDragStop script
+	TitanPluginframe:SetScript("OnDragStop", function(self)
+		if isChildButton then
+			TitanPanelButton_OnDragStop(self, true)
+		else
+			TitanPanelButton_OnDragStop(self);
+		end
+	end)
+end
+
+--[[ API
+NAME: TitanPanelButton_OnShow
+DESC: Handle the OnShow event of the requested Titan plugin.
+VAR:self - frame reference of the plugin
+--]]
+function TitanPanelButton_OnShow(self) -- Used by plugins
+	local id = nil;
+	-- ensure that the 'self' passed is a valid frame reference
+	if self and self:GetName() then
+		id = TitanUtils_GetButtonID(self:GetName());
+	end
+	-- ensure that id is a valid Titan plugin
+	if (id) then
+		TitanPanelButton_UpdateButton(id, 1);
+	end
+end
+
+--[[ API
+NAME: TitanPanelButton_OnClick
+DESC: Handle the OnClick mouse event of the requested Titan plugin.
+VAR: self - frame reference of the plugin
+VAR: button - mouse button that was clicked
+VAR: isChildButton - boolean
+NOTE:
+- Only the left and right mouse buttons are handled by Titan.
+:NOTE
+--]]
+function TitanPanelButton_OnClick(self, button, isChildButton) -- Used by plugins
+	local id
+	-- ensure that the 'self' passed is a valid frame reference
+	if self and self:GetName() then
+		id = TitanUtils_Ternary(isChildButton,
+			TitanUtils_GetParentButtonID(self:GetName()),
+			TitanUtils_GetButtonID(self:GetName()));
+	end
+
+	if id then
+		local controlFrame = TitanUtils_GetControlFrame(id);
+		local rightClickMenu = _G["TitanPanelRightClickMenu"];
+
+		if (button == "LeftButton") then
+			local isControlFrameShown;
+			if (not controlFrame) then
+				isControlFrameShown = false;
+			elseif (controlFrame:IsVisible()) then
+				isControlFrameShown = false;
+			else
+				isControlFrameShown = true;
+			end
+
+			TitanUtils_CloseAllControlFrames();
+			TitanPanelRightClickMenu_Close();
+
+			local position = TitanUtils_GetWhichBar(id)
+			local scale = TitanPanelGetVar("Scale");
+			if (isControlFrameShown) then
+				local buttonCenter = (self:GetLeft() + self:GetRight()) / 2 * scale;
+				local controlFrameRight = buttonCenter + controlFrame:GetWidth() / 2;
+				local y_off = TITAN_PANEL_BAR_HEIGHT * scale
+				if ( position == TITAN_PANEL_PLACE_TOP ) then
+					controlFrame:ClearAllPoints();
+					controlFrame:SetPoint("TOP", "UIParent", "TOPLEFT", buttonCenter, -y_off);
+
+					-- Adjust control frame position if it's off the screen
+					local offscreenX, offscreenY = TitanUtils_GetOffscreen(controlFrame);
+					if ( offscreenX == -1 ) then
+						controlFrame:ClearAllPoints();
+						controlFrame:SetPoint("TOPLEFT", "UIParent", "TOPLEFT", 0, -y_off);
+					elseif ( offscreenX == 1 ) then
+						controlFrame:ClearAllPoints();
+						controlFrame:SetPoint("TOPRIGHT", "UIParent", "TOPRIGHT", 0, -y_off);
+					end
+				else
+					controlFrame:ClearAllPoints();
+					controlFrame:SetPoint("BOTTOM", "UIParent", "BOTTOMLEFT", buttonCenter, y_off);
+
+					-- Adjust control frame position if it's off the screen
+					local offscreenX, offscreenY = TitanUtils_GetOffscreen(controlFrame);
+					if ( offscreenX == -1 ) then
+						controlFrame:ClearAllPoints();
+						controlFrame:SetPoint("BOTTOMLEFT", "UIParent", "BOTTOMLEFT", 0, y_off);
+					elseif ( offscreenX == 1 ) then
+						controlFrame:ClearAllPoints();
+						controlFrame:SetPoint("BOTTOMRIGHT", "UIParent", "BOTTOMRIGHT", 0, y_off);
+					end
+				end
+
+				controlFrame:Show();
+			end
+		elseif (button == "RightButton") then
+			TitanUtils_CloseAllControlFrames();
+			-- Show RightClickMenu anyway
+			TitanPanelRightClickMenu_Close();
+			TitanPanelRightClickMenu_Toggle(self, isChildButton);
+		end
+
+		GameTooltip:Hide();
+	end
+end
+
+--[[ API
+NAME: TitanPanelButton_OnEnter
+DESC: Handle the OnEnter cursor event of the requested Titan plugin.
+VAR: self - frame reference of the plugin
+VAR: isChildButton - boolean
+NOTE:
+- The cursor has moved over the plugin so show the plugin tooltip.
+- Save same hassle by doing nothing if the tooltip is already shown or if the cursor is moving.
+- If the "is moving" is set the user is dragging this plugin around so do nothing here.
+:NOTE
+--]]
+function TitanPanelButton_OnEnter(self, isChildButton) -- Used by plugins
+	local id = nil;
+	-- ensure that the 'self' passed is a valid frame reference
+	if self and self:GetName() then
+		id = TitanUtils_Ternary(isChildButton,
+			TitanUtils_GetParentButtonID(self:GetName()),
+			TitanUtils_GetButtonID(self:GetName()));
+	end
+
+	if (id) then
+		local controlFrame = TitanUtils_GetControlFrame(id);
+		if (controlFrame and controlFrame:IsVisible()) then
+			return;
+		elseif (TitanPanelRightClickMenu_IsVisible()) then
+			return;
+		else
+			if TITAN_PANEL_MOVING == 0 then
+				TitanPanelButton_SetTooltip(self, id);
+			end
+			if self.isMoving then
+				GameTooltip:Hide();
+			end
+		end
+	end
+end
+
+--[[ API
+NAME: TitanPanelButton_OnLeave
+DESC: Handle the OnLeave cursor event of the requested Titan plugin.
+VAR: self - frame reference of the plugin
+VAR: isChildButton - boolean
+NOTE:
+- The cursor has moved off the plugin so hide the plugin tooltip.
+:NOTE
+--]]
+function TitanPanelButton_OnLeave(self, isChildButton)
+	local id = nil;
+	-- ensure that the 'self' passed is a valid frame reference
+	if self and self:GetName() then
+		id = TitanUtils_Ternary(isChildButton,
+			TitanUtils_GetParentButtonID(self:GetName()),
+			TitanUtils_GetButtonID(self:GetName()));
+	end
+
+	if (id) then
+		GameTooltip:Hide();
+	end
+
+	if not TitanPanelGetVar("DisableTooltipFont") then
+		-- reset original Tooltip Scale
+		GameTooltip:SetScale(TitanTooltipOrigScale);
+		TitanTooltipScaleSet = 0;
+	end
+end
+
+-- local routines for Update Button
+--[[ local
+NAME: TitanPanelButton_SetButtonText
+DESC: Set / update the text of the given Titan plugin.
+VAR: id - string name of the plugin
+NOTE:
+- The plugin is expected to tell Titan what routine is to be called in <self>.registry.buttonTextFunction.
+- The text routine is called in protected mode (pcall) to ensure the Titan main routines still run.
+:NOTE
+--]]
+local format_with_label = { [0] = "" }
+for idx = 1, 4 do format_with_label[idx] = "%s%s" .. (TITAN_PANEL_LABEL_SEPARATOR .. "%s%s"):rep(idx - 1) end
+local function TitanPanelButton_SetButtonText(id)
+	if not (id and TitanUtils_IsPluginRegistered(id)) then return end
+
+	local buttonTextFunction = _G[TitanUtils_GetPlugin(id).buttonTextFunction];
+	if not buttonTextFunction then return end
+	local button = TitanUtils_GetButton(id);
+	local buttonText = _G[button:GetName()..TITAN_PANEL_TEXT];
+
+	local newfont = media:Fetch("font", TitanPanelGetVar("FontName"))
+	if newfont then
+		buttonText:SetFont(newfont, TitanPanelGetVar("FontSize"))
+	end
+
+	-- We'll be paranoid here and call the button text in protected mode.
+	-- In case the button text fails it will not take Titan with it...
+	local call_success, -- for pcall
+		label1, value1, label2, value2, label3, value3, label4, value4 =
+			pcall(buttonTextFunction, id)
+
+	if not call_success then buttonText:SetText("<?>") return end
+
+	if label1 and
+		not (label2 or label3 or label4
+		or value1 or value2 or value3 or value4) then
+		buttonText:SetText(label1)
+		return
+	end
+
+	local show_label = TitanGetVar(id, "ShowLabelText")
+	local values = 0
+	if label1 or value1 then
+		values = 1
+		if not show_label then label1 = "" end
+		if label2 or value2 then
+			values = 2
+			if not show_label then label2 = "" end
+			if label3 or value3 then
+				values = 3
+				if not show_label then label3 = "" end
+				if label4 or value4 then
+					values = 4
+					if not show_label then label4 = "" end
+				end
+			end
+		end
+	end
+
+	buttonText:SetFormattedText(format_with_label[values],
+		label1 or "", value1 or "",
+		label2 or "", value2 or "",
+		label3 or "", value3 or "",
+		label4 or "", value4 or ""
+	)
+end
+
+--[[ local
+NAME: TitanPanelButton_SetTextButtonWidth
+DESC: Set the text width of the given Titan plugin that is text only.
+VAR: id - string name of the plugin
+VAR: setButtonWidth - new width
+NOTE:
+- Titan uses a tolerance setting to prevent endless updating of the text width.
+:NOTE
+--]]
+local function TitanPanelButton_SetTextButtonWidth(id, setButtonWidth)
+	if (id) then
+		local button = TitanUtils_GetButton(id);
+		local text = _G[button:GetName()..TITAN_PANEL_TEXT];
+		if ( setButtonWidth
+		or button:GetWidth() == 0
+		or button:GetWidth() - text:GetWidth() > TITAN_PANEL_BUTTON_WIDTH_CHANGE_TOLERANCE
+		or button:GetWidth() - text:GetWidth() < -TITAN_PANEL_BUTTON_WIDTH_CHANGE_TOLERANCE ) then
+			button:SetWidth(text:GetWidth());
+			TitanPanelButton_Justify();
+		end
+	end
+end
+
+--[[ local
+NAME: TitanPanelButton_SetIconButtonWidth
+DESC: Set the icon width of the given Titan plugin that is icon only.
+VAR: id - string name of the plugin
+NOTE:
+- The plugin is expected to tell Titan what the icon width is in <self>.registry.iconButtonWidth.
+:NOTE
+--]]
+local function TitanPanelButton_SetIconButtonWidth(id)
+	if (id) then
+		local button = TitanUtils_GetButton(id);
+		if ( TitanUtils_GetPlugin(id).iconButtonWidth ) then
+			button:SetWidth(TitanUtils_GetPlugin(id).iconButtonWidth);
+		end
+	end
+end
+
+--[[ local
+NAME: TitanPanelButton_SetComboButtonWidth
+DESC: Set the icon width of the given Titan plugin that is a combo - icon & text.
+VAR: id - string name of the plugin
+VAR: setButtonWidth - new width
+NOTE:
+- The plugin is expected to tell Titan what the icon width is in <self>.registry.iconButtonWidth.
+:NOTE
+--]]
+local function TitanPanelButton_SetComboButtonWidth(id, setButtonWidth)
+	if (id) then
+		local button = TitanUtils_GetButton(id)
+		if not button then return end -- sanity check
+
+		local text = _G[button:GetName()..TITAN_PANEL_TEXT];
+		local icon = _G[button:GetName().."Icon"];
+		local iconWidth, iconButtonWidth, newButtonWidth;
+
+		-- Get icon button width
+		iconButtonWidth = 0;
+		if ( TitanUtils_GetPlugin(id).iconButtonWidth ) then
+			iconButtonWidth = TitanUtils_GetPlugin(id).iconButtonWidth;
+		elseif ( icon:GetWidth() ) then
+			iconButtonWidth = icon:GetWidth();
+		end
+
+		if ( TitanGetVar(id, "ShowIcon") and ( iconButtonWidth ~= 0 ) ) then
+			icon:Show();
+			text:ClearAllPoints();
+			text:SetPoint("LEFT", icon:GetName(), "RIGHT", 2, 1);
+
+			newButtonWidth = text:GetWidth() + iconButtonWidth;
+		else
+			icon:Hide();
+			text:ClearAllPoints();
+			text:SetPoint("LEFT", button:GetName(), "LEFT", 0, 1);
+
+			newButtonWidth = text:GetWidth();
+		end
+
+		if ( setButtonWidth
+		or button:GetWidth() == 0
+		or button:GetWidth() - newButtonWidth > TITAN_PANEL_BUTTON_WIDTH_CHANGE_TOLERANCE
+		or button:GetWidth() - newButtonWidth < -TITAN_PANEL_BUTTON_WIDTH_CHANGE_TOLERANCE )
+		then
+			button:SetWidth(newButtonWidth);
+			TitanPanelButton_Justify();
+		end
+	end
+end
+
+--[[ API
+NAME: TitanPanelButton_UpdateButton
+DESC: Update the display of the given Titan plugin.
+VAR: id - string name of the plugin
+VAR: setButtonWidth - new width
+--]]
+function TitanPanelButton_UpdateButton(id, setButtonWidth)  -- Used by plugins
+	local button, id = TitanUtils_GetButton(id);
+	local plugin = TitanUtils_GetPlugin(id)
+	-- safeguard to avoid errors
+	if not TitanUtils_IsPluginRegistered(id) then return end
+
+	if ( TitanPanelButton_IsText(id) ) then
+		-- Update textButton
+		TitanPanelButton_SetButtonText(id);
+		TitanPanelButton_SetTextButtonWidth(id, setButtonWidth);
+
+	elseif ( TitanPanelButton_IsIcon(id) ) then
+		-- Update iconButton
+		TitanPanelButton_SetButtonIcon(id, (plugin.iconCoords or nil),
+			(plugin.iconR or nil),(plugin.iconG or nil),(plugin.iconB or nil)
+			);
+		TitanPanelButton_SetIconButtonWidth(id);
+
+	elseif ( TitanPanelButton_IsCombo(id) ) then
+		-- Update comboButton
+		TitanPanelButton_SetButtonText(id);
+		TitanPanelButton_SetButtonIcon(id, (plugin.iconCoords or nil),
+			(plugin.iconR or nil),(plugin.iconG or nil),(plugin.iconB or nil)
+			);
+		TitanPanelButton_SetComboButtonWidth(id, setButtonWidth);
+	end
+end
+
+--[[ API
+NAME: TitanPanelButton_UpdateTooltip
+DESC: Update the tooltip of the given Titan plugin.
+VAR: self - frame reference of the plugin
+--]]
+function TitanPanelButton_UpdateTooltip(self) -- Used by plugins
+	if not self then return end
+	if (GameTooltip:IsOwned(self)) then
+		local id = TitanUtils_GetButtonID(self:GetName());
+		TitanPanelButton_SetTooltip(self, id);
+	end
+end
+
+--[[ API
+NAME: TitanPanelButton_SetButtonIcon
+DESC: Set the icon of the given Titan plugin.
+VAR: id - string name of the plugin
+VAR: iconCoords - if given, this is the placing of the icon within the plugin
+VAR: iconR - if given, this is the Red (RBG) setting of the icon
+VAR: iconG - if given, this is the Green (RBG) setting of the icon
+VAR: iconB - if given, this is the Blue (RBG) setting of the icon
+--]]
+function TitanPanelButton_SetButtonIcon(id, iconCoords, iconR, iconG, iconB)
+	if (id and TitanUtils_IsPluginRegistered(id)) then
+		local button = TitanUtils_GetButton(id);
+		local icon = _G[button:GetName().."Icon"];
+		local iconTexture = TitanUtils_GetPlugin(id).icon;
+		local iconWidth = TitanUtils_GetPlugin(id).iconWidth;
+
+		if (iconTexture) and icon then
+			icon:SetTexture(iconTexture);
+		end
+		if (iconWidth) and icon then
+			icon:SetWidth(iconWidth);
+		end
+
+		-- support for iconCoords, iconR, iconG, iconB attributes
+		if iconCoords and icon then
+			icon:SetTexCoord(unpack(iconCoords))
+		end
+		if iconR and iconG and iconB and icon then
+			icon:SetVertexColor(iconR, iconG, iconB)
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanPanelButton_GetType
+DESC: Get the type of the given Titan plugin.
+VAR: id - string name of the plugin
+OUT: type - The type of the plugin (text, icon, combo (default))
+NOTE:
+- This assumes that the developer is playing nice and is using the Titan templates as is...
+:NOTE
+--]]
+function TitanPanelButton_GetType(id)
+	-- id is required
+	if (not id) then
+		return;
+	end
+
+	local button = TitanUtils_GetButton(id);
+	local type;
+	if button then
+		local text = _G[button:GetName()..TITAN_PANEL_TEXT];
+		local icon = _G[button:GetName().."Icon"];
+
+		if (text and icon) then
+			type = TITAN_PANEL_BUTTON_TYPE_COMBO;
+		elseif (text and not icon) then
+			type = TITAN_PANEL_BUTTON_TYPE_TEXT;
+		elseif (not text and icon) then
+			type = TITAN_PANEL_BUTTON_TYPE_ICON;
+		elseif (not text and not icon) then
+			type = TITAN_PANEL_BUTTON_TYPE_CUSTOM;
+		end
+	else
+		type = TITAN_PANEL_BUTTON_TYPE_COMBO;
+	end
+
+	return type;
+end
+
diff --git a/Titan/TitanPanelTemplate.xml b/Titan/TitanPanelTemplate.xml
new file mode 100644
index 0000000..1f857c7
--- /dev/null
+++ b/Titan/TitanPanelTemplate.xml
@@ -0,0 +1,157 @@
+<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="TitanPanelTemplate.lua" />
+	<Button name="TitanPanelButtonTemplate" hidden="true" movable="true" virtual="true">
+		<Frames>
+			<Frame name="$parentRightClickMenu" inherits="L_UIDropDownMenuTemplate" id="1" hidden="true">
+			</Frame>
+		</Frames>
+		<Scripts>
+			<OnLoad>
+				TitanPanelButton_OnLoad(self);
+			</OnLoad>
+			<OnShow>
+				TitanPanelButton_OnShow(self);
+			</OnShow>
+			<OnClick>
+				TitanPanelButton_OnClick(self, button);
+			</OnClick>
+			<OnEnter>
+				TitanPanelButton_OnEnter(self);
+			</OnEnter>
+			<OnLeave>
+				TitanPanelButton_OnLeave(self);
+			</OnLeave>
+		</Scripts>
+	</Button>
+	<Button name="TitanPanelChildButtonTemplate" hidden="true" virtual="true">
+		<Frames>
+			<Frame name="$parentRightClickMenu" inherits="L_UIDropDownMenuTemplate" id="1" hidden="true">
+			</Frame>
+		</Frames>
+		<Scripts>
+			<OnLoad>
+				TitanPanelButton_OnLoad(self, true);
+			</OnLoad>
+			<OnClick>
+				TitanPanelButton_OnClick(self, button, true);
+			</OnClick>
+			<OnEnter>
+				TitanPanelButton_OnEnter(self, true);
+			</OnEnter>
+			<OnLeave>
+				TitanPanelButton_OnLeave(self, true);
+			</OnLeave>
+		</Scripts>
+	</Button>
+	<Button name="TitanPanelTextTemplate" inherits="TitanPanelButtonTemplate" virtual="true">
+		<Size>
+			<AbsDimension x="0" y="16" />
+		</Size>
+		<ButtonText name="$parentText" inherits="GameFontNormalSmall" justifyH="LEFT">
+			<Anchors>
+				<Anchor point="LEFT">
+					<Offset>
+						<AbsDimension x="0" y="1" />
+					</Offset>
+				</Anchor>
+			</Anchors>
+		</ButtonText>
+		<PushedTextOffset>
+			<AbsDimension x="0" y="0"/>
+		</PushedTextOffset>
+	</Button>
+	<Button name="TitanPanelIconTemplate" inherits="TitanPanelButtonTemplate" virtual="true">
+		<Size>
+			<AbsDimension x="16" y="16" />
+		</Size>
+		<Layers>
+			<Layer level="ARTWORK">
+				<Texture name="$parentIcon">
+					<Size>
+						<AbsDimension x="16" y="16"/>
+					</Size>
+					<Anchors>
+						<Anchor point="LEFT" />
+					</Anchors>
+				</Texture>
+			</Layer>
+		</Layers>
+	</Button>
+	<Button name="TitanPanelComboTemplate" inherits="TitanPanelButtonTemplate" virtual="true">
+		<Size>
+			<AbsDimension x="0" y="16" />
+		</Size>
+		<Layers>
+			<Layer level="ARTWORK">
+				<Texture name="$parentIcon">
+					<Size>
+						<AbsDimension x="0" y="16"/>
+					</Size>
+					<Anchors>
+						<Anchor point="LEFT" />
+					</Anchors>
+				</Texture>
+			</Layer>
+		</Layers>
+		<ButtonText name="$parentText" inherits="GameFontNormalSmall" justifyH="LEFT" />
+		<PushedTextOffset>
+			<AbsDimension x="0" y="0"/>
+		</PushedTextOffset>
+	</Button>
+	<Slider name="TitanOptionsSliderTemplate" orientation="VERTICAL" virtual="true" enableMouse="true" EnableMouseWheel="true">
+		<Size>
+			<AbsDimension x="10" y="100"/>
+		</Size>
+		<HitRectInsets>
+			<AbsInset left="-10" right="-10" top="0" bottom="0"/>
+		</HitRectInsets>
+		<Backdrop bgFile="Interface\Buttons\UI-SliderBar-Background" edgeFile="Interface\Buttons\UI-SliderBar-Border" tile="true">
+			<EdgeSize>
+				<AbsValue val="8"/>
+			</EdgeSize>
+			<TileSize>
+				<AbsValue val="8"/>
+			</TileSize>
+			<BackgroundInsets>
+				<AbsInset left="6" right="6" top="3" bottom="3"/>
+			</BackgroundInsets>
+		</Backdrop>
+		<Layers>
+			<Layer level="ARTWORK">
+				<FontString name="$parentText" inherits="GameFontGreenSmall">
+					<Anchors>
+						<Anchor point="LEFT" relativePoint="RIGHT">
+							<Offset>
+								<AbsDimension x="2" y="0" />
+							</Offset>
+						</Anchor>
+					</Anchors>
+				</FontString>
+				<FontString name="$parentLow" inherits="GameFontHighlightSmall" text="LOW">
+					<Anchors>
+						<Anchor point="BOTTOM" relativePoint="TOP">
+							<Offset>
+								<AbsDimension x="0" y="3"/>
+							</Offset>
+						</Anchor>
+					</Anchors>
+				</FontString>
+				<FontString name="$parentHigh" inherits="GameFontHighlightSmall" text="HIGH">
+					<Anchors>
+						<Anchor point="TOP" relativePoint="BOTTOM">
+							<Offset>
+								<AbsDimension x="0" y="0"/>
+							</Offset>
+						</Anchor>
+					</Anchors>
+				</FontString>
+			</Layer>
+		</Layers>
+		<ThumbTexture name="$parentThumb" file="Interface\Buttons\UI-SliderBar-Button-Vertical">
+			<Size>
+				<AbsDimension x="32" y="32"/>
+			</Size>
+		</ThumbTexture>
+	</Slider>
+</Ui>
diff --git a/Titan/TitanPluginDoc.lua b/Titan/TitanPluginDoc.lua
new file mode 100644
index 0000000..2471c3e
--- /dev/null
+++ b/Titan/TitanPluginDoc.lua
@@ -0,0 +1,55 @@
+--[[ File
+NAME: DocTitanPlugin.lua
+DESC: This file contains documentation of Titan to assist a developer.
+:DESC
+--]]
+--[[ API
+NAME: Titan API overview for developers
+DESC: Updated Dec 2011
+
+This documentation is intended for addon developers that wish to create a Titan plugin.
+
+Terms:
+Throughout the documentation and naming of routines and variables there are some terms that are used interchangably.
+- Plugin / button / frame
+- Character / player / toon
+- Plugin ID / plugin name
+Over time we desire to consolidate terms but it may take time.
+
+Plugin Types:
+Titan allows two types of plugins:
+- Titan native
+- LDB enabled addons
+See http://code.google.com/p/titanpanel/downloads/list to download examples.
+
+Titan Plugin Recognition:
+Titan native plugins must use one of the Titan templates found in TitanPanelTemplates.xml.
+
+LDB Plugin Recognition:
+LDB enabled addons only need to adhere to the LDB (Lib Data Broker) 1.1 spec.
+Titan uses the callback feature ("LibDataBroker_DataObjectCreated") to recognize LDB addons.
+At player_login Titan registers the call back.
+It then loops through the LDB objects using :DataObjectIterator().
+When any LDB object is found Titan will attempt to create a Titan native plugin for display.
+
+Registration Steps:
+Titan attempts to register each plugin it recognizes. Registration for Titan is a two step process.
+Step one: On the creation of a Titan plugin frame place the plugin in a holding table until the "Player Entering World" (PEW) event is fired.
+Step two: Once Titan is initialized it will take each plugin from this table and attempt to register it for display. The attempt uses a protected call (pcall) to protect the Titan core and (hopefully) prevents Titan from crashing.
+
+Registration Attempts:
+Each plugin attempt is placed in an 'attempted' table along with the results of the registration. This attempted table is accessible to the user in the Titan "Attempted" options. The developer can see what happened and users can report issues using that data.
+
+Registry Table of Each Plugin:
+Each plugin must contain a table called "registry".
+   self.registry = {}
+
+The registry table must have a unique id across all Titan plugins that are loaded.
+   self.registry = {id = "MyPlugin"}
+This is all that is required for successful registration. It will register but will be a rather dull plugin.
+It is strongly recommended that you download the TitanStarter plugin example(see the link above). The example explains all the elements of the registry that Titan uses. The example is based on TitanBags so you can play with the code and compare it to a supported plugin.
+
+Titan API:
+The functions that Titan offers to the plugin developer are in a separate document. Within the Titan files you can recognize an API routine by the comment just before the function declaration. The comment block will have "API" as part of the starting block.
+:DESC
+--]]
\ No newline at end of file
diff --git a/Titan/TitanUtils.lua b/Titan/TitanUtils.lua
new file mode 100644
index 0000000..06c7085
--- /dev/null
+++ b/Titan/TitanUtils.lua
@@ -0,0 +1,1994 @@
+--[[ File
+NAME: TitanUtils.lua
+DESC: This file contains various utility routines used by Titan and routines available to plugin developers.
+--]]
+TITAN_ID = "Titan"
+TITAN_DEBUG_ARRAY_MAX = 100
+TITAN_PANEL_NONMOVABLE_PLUGINS = {};
+TITAN_PANEL_MENU_FUNC_HIDE = "TitanPanelRightClickMenu_Hide";
+TitanPlugins = {};  -- Used by plugins
+TitanPluginsIndex = {};
+TITAN_NOT_REGISTERED = _G["RED_FONT_COLOR_CODE"].."Not_Registered_Yet".._G["FONT_COLOR_CODE_CLOSE"]
+TITAN_REGISTERED = _G["GREEN_FONT_COLOR_CODE"].."Registered".._G["FONT_COLOR_CODE_CLOSE"]
+TITAN_REGISTER_FAILED = _G["RED_FONT_COLOR_CODE"].."Failed_to_Register".._G["FONT_COLOR_CODE_CLOSE"]
+
+local _G = getfenv(0);
+local L = LibStub("AceLocale-3.0"):GetLocale(TITAN_ID, true)
+local media = LibStub("LibSharedMedia-3.0")
+
+--
+-- The routines labeled API are useable by addon developers
+--
+
+--[[ API
+NAME: TitanUtils_GetBarAnchors
+DESC: Get the anchors of the bottom most top bar and the top most bottom bar.
+   Intended for addons that modify the UI so they can adjust for Titan.
+   The anchors adjust depending on what Titan bars the user displays.
+:DESC
+VAR:  None
+OUT: frame - TitanPanelTopAnchor frame reference - Titan uses the space ABOVE this
+OUT: frame - TitanPanelBottomAnchor frame reference - Titan uses the space BELOW this
+NOTE:
+-  The two anchors are implemented as 2 frames that are moved by Titan depending on which bars are shown.
+:NOTE
+--]]
+function TitanUtils_GetBarAnchors() -- Used by addons
+	return TitanPanelTopAnchor, TitanPanelBottomAnchor
+end
+
+--[[ API
+NAME: TitanUtils_GetMinimapAdjust
+DESC: Return the current setting of the Titan MinimapAdjust option.
+VAR: None
+OUT: The value of the MinimapAdjust option
+--]]
+function TitanUtils_GetMinimapAdjust() -- Used by addons
+	return not TitanPanelGetVar("MinimapAdjust")
+end
+
+--[[ API
+NAME: TitanUtils_SetMinimapAdjust
+DESC: Set the current setting of the Titan MinimapAdjust option.
+VAR:  bool - true (off) or false (on)
+OUT:  None
+--]]
+function TitanUtils_SetMinimapAdjust(bool) -- Used by addons
+	-- This routine allows an addon to turn on or off
+	-- the Titan minimap adjust.
+	TitanPanelSetVar("MinimapAdjust", not bool)
+end
+
+--[[ API
+NAME: TitanUtils_AddonAdjust
+DESC: Tell Titan to adjust (or not) a frame.
+VAR: frame - is the name (string) of the frame
+VAR: bool  - true if the addon will adjust the frame or false if Titan will adjust
+OUT:  None
+Note:
+- Titan will NOT store the adjust value across a log out / exit.
+- This is a generic way for an addon to tell Titan to not adjust a frame. The addon will take responsibility for adjusting that frame. This is useful for UI style addons so the user can run Titan and a modifed UI.
+- The list of frames Titan adjusts is specified in TitanMovableData within TitanMovable.lua.
+- If the frame name is not in TitanMovableData then Titan does not adjust that frame.
+:NOTE
+--]]
+function TitanUtils_AddonAdjust(frame, bool) -- Used by addons
+	TitanMovable_AddonAdjust(frame, bool)
+end
+
+--------------------------------------------------------------
+--
+-- Plugin button search & manipulation routines
+--
+--[[ API
+NAME: TitanUtils_GetButton
+DESC: Return the actual button frame and the plugin id.
+VAR: id - is the id of the plugin
+OUT: frame - The button table
+OUT: string - The id of the plugin
+--]]
+function TitanUtils_GetButton(id) -- Used by plugins
+	if (id) then
+		return _G["TitanPanel"..id.."Button"], id;
+	else
+		return nil, nil;
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetRealPosition
+DESC: Return whether the plugin is on the top or bottom bar.
+VAR: id - is the id of the plugin
+OUT: bottom(2) or top(1)-default
+NOTE:
+- This returns top or bottom NOT which bar (1-4) the plugin is on.
+:NOTE
+--]]
+function TitanUtils_GetRealPosition(id) -- Used by plugins
+	-- This will return top / bottom but it is a compromise.
+	-- With the introduction of independent bars there is
+	-- more than just top / bottom.
+	-- This should work in the sense that the plugins using this
+	-- would overlap the double bar.
+	local bar = TitanUtils_GetWhichBar(id)
+	local bar_pos = nil
+	for idx,v in pairs (TitanBarData) do
+		if bar == TitanBarData[idx].name then
+			bar_pos = TitanBarData[idx].vert
+		end
+	end
+	return (bar_pos == TITAN_BOTTOM and TITAN_PANEL_PLACE_BOTTOM or TITAN_PANEL_PLACE_TOP)
+end
+
+--[[ API
+NAME: TitanUtils_GetButtonID
+DESC: Return the plugin id of the given name. This can be used to see if a plugin exists.
+VAR: name - is the id of the plugin
+OUT: string - plugin id or nil
+--]]
+function TitanUtils_GetButtonID(name)
+	if name then
+		local s, e, id = string.find(name, "TitanPanel(.*)Button");
+		return id;
+	else
+		return nil;
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetParentButtonID
+DESC: Return the plugin id of the parent of the given name, if it exists.
+VAR: name - is the id of the plugin
+OUT: string - plugin id or nil
+--]]
+function TitanUtils_GetParentButtonID(name)
+	local frame = TitanUtils_Ternary(name, _G[name], nil);
+
+	if ( frame and frame:GetParent() ) then
+		return TitanUtils_GetButtonID(frame:GetParent():GetName());
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetButtonIDFromMenu
+DESC: Return the plugin id of whatever the mouse is over. Used in the right click menu on load.
+VAR: self - is the id of the frame
+OUT: string - plugin id or nil
+NOTE:
+- The plugin id returned could be the Titan bar or a plugin or nil.
+:NOTE
+--]]
+function TitanUtils_GetButtonIDFromMenu(self)
+	local ret = nil
+	if self and self:GetParent() then
+		local name = self:GetParent():GetName()
+		local s, e, id = string.find(name, TITAN_PANEL_DISPLAY_PREFIX);
+		local temp = ""
+
+		if not s == nil then
+			-- The click was on the Titan bar itself
+			ret = "Bar";
+		elseif self:GetParent():GetParent():GetName() then
+			local pname = self:GetParent():GetParent():GetName()
+			-- TitanPanelChildButton
+			-- expecting this to be a TitanPanelChildButtonTemplate
+			temp = TitanUtils_GetButtonID(pname)
+			if temp then
+				-- should be ok
+				ret = temp
+			else
+				-- the frame container is expected to be without a name
+				-- This trips when the user right clicks a LDB plugin...
+			end
+		else
+			-- TitanPanelButton
+			temp = TitanUtils_GetButtonID(self:GetParent():GetName())
+			if temp then
+				-- should be ok
+				ret = temp
+			else
+				TitanDebug("Could not determine Titan ID for '"
+				..(self:GetParent():GetParent() or "?").."'. "
+				,"error")
+			end
+		end
+	end
+
+	return ret
+end
+
+--[[ API
+NAME: TitanUtils_GetPlugin
+DESC: Return the plugin itself (table and all).
+VAR: id - is the id of the plugin
+OUT: table - plugin or nil
+--]]
+function TitanUtils_GetPlugin(id)
+	if (id) then
+		return TitanPlugins[id];
+	else
+		return nil;
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetWhichBar
+DESC: Return the bar the plugin is shown on.
+VAR: id - is the id of the plugin
+OUT: string - bar name or nil
+--]]
+function TitanUtils_GetWhichBar(id)
+	local i = TitanPanel_GetButtonNumber(id);
+	if TitanPanelSettings.Location[i] == nil then
+		return
+	else
+		return TitanPanelSettings.Location[i];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_PickBar
+DESC: Return the first bar that is shown.
+VAR:  None
+OUT: string - bar name or nil
+--]]
+function TitanUtils_PickBar()
+	-- Pick the 'first' bar shown.
+	-- This is used for defaulting where plugins are put
+	-- if using the Titan options screen.
+	for idx,v in pairs (TitanBarOrder) do
+		if TitanPanelGetVar(v.."_Show") then
+			return v
+		end
+	end
+	-- fail safe - return something
+	return nil
+end
+
+--[[ API
+NAME: TitanUtils_ToRight
+DESC: See if the plugin is to be on the right.
+   There are 3 methods to place a plugin on the right:
+   1) DisplayOnRightSide saved variable logic (preferred)
+   2) Create a plugin button using the TitanPanelIconTemplate
+   3) Place a plugin in TITAN_PANEL_NONMOVABLE_PLUGINS (NOT preferred)
+:DESC
+VAR:  None
+OUT: bool - true or nil. true if the plugin is to be placed on the right side of a bar.
+--]]
+function TitanUtils_ToRight(id)
+	local found = nil
+	for index, _ in ipairs(TITAN_PANEL_NONMOVABLE_PLUGINS) do
+		if id == TITAN_PANEL_NONMOVABLE_PLUGINS[index] then
+			found = true;
+		end
+	end
+
+	if TitanGetVar(id, "DisplayOnRightSide")
+	or TitanPanelButton_IsIcon(id)
+	then
+		found = true
+	end
+
+	return found
+end
+
+--------------------------------------------------------------
+--
+-- General util routines
+--
+
+--[[ API
+NAME: TitanUtils_Ternary
+DESC: Return b or c if true or false respectively
+VAR: a - value to determine true or false
+VAR: b - value to use if true
+VAR: c - value to use if false or nil
+OUT: value - b (true) or c (false)
+--]]
+function TitanUtils_Ternary(a, b, c) -- Used by plugins
+	if (a) then
+		return b;
+	else
+		return c;
+	end
+end
+
+--[[ API
+NAME: TitanUtils_Toggle
+DESC: Flip the value assuming it is true or false
+VAR: value - value to start with
+OUT: bool - true or false
+--]]
+function TitanUtils_Toggle(value) -- Used by plugins
+	if (value == 1 or value == true) then
+		return nil;
+	else
+		return 1;
+	end
+end
+
+--[[ API
+NAME: TitanUtils_Min
+DESC: Return the min of a or b
+VAR: a - a value
+VAR: b - a value
+OUT:
+- value of min (a, b)
+--]]
+function TitanUtils_Min(a, b)
+	if (a and b) then
+--		return ( a < b ) and a or b
+		return TitanUtils_Ternary((a < b), a, b);
+	end
+end
+
+--[[ API
+NAME: TitanUtils_Max
+DESC: Return the max of a or b
+VAR: a - a value
+VAR: b - a value
+OUT: value - value of max (a, b)
+--]]
+function TitanUtils_Max(a, b)
+	if (a and b) then
+		return TitanUtils_Ternary((a > b), a, b);
+---		return ( a > b ) and a or b
+	end
+end
+
+--[[ local
+NAME: GetTimeParts
+DESC: Use the seconds (s) to return its parts.
+VAR: s - a time value in seconds
+OUT: int - days
+OUT: int - hours
+OUT: int - minutes
+OUT: int - seconds
+--]]
+local function GetTimeParts(s)
+	local days = 0
+	local hours = 0
+	local minutes = 0
+	local seconds = 0
+	if not s or (s < 0) then
+		seconds = L["TITAN_NA"]
+	else
+		days = floor(s/24/60/60); s = mod(s, 24*60*60);
+		hours = floor(s/60/60); s = mod(s, 60*60);
+		minutes = floor(s/60); s = mod(s, 60);
+		seconds = s;
+	end
+
+	return days, hours, minutes, seconds
+end
+
+--[[ API
+NAME: TitanUtils_GetEstTimeText
+DESC: Use the seconds (s) to return an estimated time.
+VAR: s - a time value in seconds
+OUT: string - string with localized, estimated elapsed time using spaces and leaving off the rest
+--]]
+function TitanUtils_GetEstTimeText(s)
+	local timeText = "";
+	local days, hours, minutes, seconds = GetTimeParts(s)
+	local fracdays = days + (hours/24);
+	local frachours = hours + (minutes/60);
+	if seconds == L["TITAN_NA"] then
+		timeText = L["TITAN_NA"];
+	else
+		if (days ~= 0) then
+			timeText = timeText..format("%4.1f"..L["TITAN_DAYS_ABBR"].." ", fracdays);
+		elseif (days ~= 0 or hours ~= 0) then
+			timeText = timeText..format("%4.1f"..L["TITAN_HOURS_ABBR"].." ", frachours);
+		elseif (days ~= 0 or hours ~= 0 or minutes ~= 0) then
+			timeText = timeText..format("%d"..L["TITAN_MINUTES_ABBR"].." ", minutes);
+		else
+			timeText = timeText..format("%d"..L["TITAN_SECONDS_ABBR"], seconds);
+		end
+	end
+	return timeText;
+end
+
+--[[ API
+NAME: TitanUtils_GetFullTimeText
+DESC: break the seconds (s) into days, hours, minutes, and seconds
+VAR: s - a time value in seconds
+OUT: string - string with localized days, hours, minutes, and seconds using commas and including zeroes
+--]]
+function TitanUtils_GetFullTimeText(s)
+	local days, hours, minutes, seconds = GetTimeParts(s)
+	if seconds == L["TITAN_NA"] then
+		return L["TITAN_NA"];
+	else
+		return format("%d"..L["TITAN_DAYS_ABBR"]
+			..", %2d"..L["TITAN_HOURS_ABBR"]
+			..", %2d"..L["TITAN_MINUTES_ABBR"]
+			..", %2d"..L["TITAN_SECONDS_ABBR"],
+				days, hours, minutes, seconds);
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetAbbrTimeText
+DESC: break the seconds (s) into days, hours, minutes, and seconds
+VAR: s - a time value in seconds
+OUT: string - string with localized days, hours, minutes, and seconds using spaces and including zeroes
+--]]
+function TitanUtils_GetAbbrTimeText(s) -- Used by plugins
+	local timeText = "";
+	local days, hours, minutes, seconds = GetTimeParts(s)
+	if seconds == L["TITAN_NA"] then
+		timeText = L["TITAN_NA"];
+	else
+		if (days ~= 0) then
+			timeText = timeText..format("%d"..L["TITAN_DAYS_ABBR"].." ", days);
+		end
+		if (days ~= 0 or hours ~= 0) then
+			timeText = timeText..format("%d"..L["TITAN_HOURS_ABBR"].." ", hours);
+		end
+		if (days ~= 0 or hours ~= 0 or minutes ~= 0) then
+			timeText = timeText..format("%d"..L["TITAN_MINUTES_ABBR"].." ", minutes);
+		end
+		timeText = timeText..format("%d"..L["TITAN_SECONDS_ABBR"], seconds);
+	end
+	return timeText;
+end
+
+--[[ API
+NAME: TitanUtils_GetControlFrame
+DESC: return the control frame, if one was created.
+VAR: id - id of the plugin
+OUT: frame - nil or the control frame
+NOTE
+- This may not be used anymore.
+:NOTE
+--]]
+function TitanUtils_GetControlFrame(id)
+	if (id) then
+		return _G["TitanPanel"..id.."ControlFrame"];
+	else
+		return nil;
+	end
+end
+
+--[[ API
+NAME: TitanUtils_TableContainsValue
+DESC: Determine if the table contains the value.
+VAR: table - table to search
+VAR: value - value to find
+OUT: int - nil or the index to value
+--]]
+function TitanUtils_TableContainsValue(table, value)
+	if (table and value) then
+		for i, v in pairs(table) do
+			if (v == value) then
+				return i;
+			end
+		end
+	end
+end
+
+--[[ API
+NAME: TitanUtils_TableContainsIndex
+DESC: Determine if the table contains the index.
+VAR: table - table to search
+VAR: index - index to find
+OUT: int - nil or the index
+--]]
+function TitanUtils_TableContainsIndex(table, index)
+	if (table and index and table[index] ~= nil) then
+		return index
+	end
+--[[
+	if (table and index) then
+		for i, v in pairs(table) do
+			if (i == index) then
+				return i;
+			end
+		end
+	end
+--]]
+end
+
+--[[ API
+NAME: TitanUtils_GetCurrentIndex
+DESC: Determine if the table contains the value.
+VAR: table - table to search
+VAR: value - value to find
+OUT: int - nil or the index to value
+--]]
+function TitanUtils_GetCurrentIndex(table, value)
+	return TitanUtils_TableContainsValue(table, value);
+end
+
+--[[ API
+NAME: TitanUtils_PrintArray
+DESC: Debug tool that will attempt to output the index and value of the array passed in.
+VAR: array - array to output
+OUT: table - Array output to the chat window
+--]]
+function TitanUtils_PrintArray(array)
+	if (not array) then
+		TitanDebug("array is nil");
+	else
+		TitanDebug("{");
+		for i, v in array do
+			TitanDebug("array[" .. tostring(i) .. "] = " .. tostring(v));
+		end
+		TitanDebug("}");
+	end
+
+end
+
+--[[ API
+NAME: TitanUtils_GetRedText
+DESC: Make the given text red.
+VAR: text - text to color
+OUT: string - Red string with proper start and end font encoding
+--]]
+function TitanUtils_GetRedText(text) -- Used by plugins
+	if (text) then
+		return _G["RED_FONT_COLOR_CODE"]..text.._G["FONT_COLOR_CODE_CLOSE"];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetGoldText
+DESC: Make the given text gold.
+VAR: text - text to color
+OUT: string - Gold string with proper start and end font encoding
+--]]
+function TitanUtils_GetGoldText(text)
+	if (text) then
+		return "|cffffd700"..text.._G["FONT_COLOR_CODE_CLOSE"];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetGreenText
+DESC: Make the given text green.
+VAR: text - text to color
+OUT: string - Green string with proper start and end font encoding
+--]]
+function TitanUtils_GetGreenText(text) -- Used by plugins
+	if (text) then
+		return _G["GREEN_FONT_COLOR_CODE"]..text.._G["FONT_COLOR_CODE_CLOSE"];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetBlueText
+DESC: Make the given text blue.
+VAR: text - text to color
+OUT: string - Blue string with proper start and end font encoding
+--]]
+function TitanUtils_GetBlueText(text)
+	if (text) then
+		return "|cff0000ff"..text.._G["FONT_COLOR_CODE_CLOSE"];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetNormalText
+DESC: Make the given text normal (gray-white).
+VAR: text - text to color
+OUT: string - Normal string with proper start and end font encoding
+--]]
+function TitanUtils_GetNormalText(text) -- Used by plugins
+	if (text) then
+		return _G["NORMAL_FONT_COLOR_CODE"]..text.._G["FONT_COLOR_CODE_CLOSE"];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetHighlightText
+DESC: Make the given text highlight (brighter white).
+VAR: text - text to color
+OUT: string - Highlight string with proper start and end font encoding
+--]]
+function TitanUtils_GetHighlightText(text) -- Used by plugins
+	if (text) then
+		return _G["HIGHLIGHT_FONT_COLOR_CODE"]..text.._G["FONT_COLOR_CODE_CLOSE"];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetColoredText
+DESC: Make the given text a custom color.
+VAR: text - text to color
+VAR: color - color is the color table with r, b, g values set.
+OUT: string - Custom color string with proper start and end font encoding
+--]]
+function TitanUtils_GetColoredText(text, color) -- Used by plugins
+	if (text and color) then
+		local redColorCode = format("%02x", color.r * 255);
+		local greenColorCode = format("%02x", color.g * 255);
+		local blueColorCode = format("%02x", color.b * 255);
+		local colorCode = "|cff"..redColorCode..greenColorCode..blueColorCode;
+		return colorCode..text.._G["FONT_COLOR_CODE_CLOSE"];
+	end
+end
+
+--[[ API
+NAME: TitanUtils_GetThresholdColor
+DESC: Flexable routine that returns the threshold color for a given value using a table as input.
+VAR: ThresholdTable - table holding the list of colors and values
+VAR: value -
+OUT: string - The color value from the treshhold table
+--]]
+function TitanUtils_GetThresholdColor(ThresholdTable, value)
+	if ( not tonumber(value) or type(ThresholdTable) ~= "table"
+	or ThresholdTable.Values == nil or ThresholdTable.Colors == nil
+	or table.getn(ThresholdTable.Values) >= table.getn(ThresholdTable.Colors)
+	) then
+		return _G["GRAY_FONT_COLOR"];
+	end
+
+	local n = table.getn(ThresholdTable.Values) + 1;
+	for i = 1, n do
+		local low = TitanUtils_Ternary(i == 1, nil, ThresholdTable.Values[i-1]); -- lowest
+		local high = TitanUtils_Ternary(i == n, nil, ThresholdTable.Values[i]);  -- highest
+
+		if ( not low and not high ) then
+			-- No threshold values
+			return ThresholdTable.Colors[i];
+
+		elseif ( not low and high ) then
+			-- Value is smaller than the first threshold
+			if ( value < high ) then return ThresholdTable.Colors[i] end
+
+		elseif ( low and not high ) then
+			-- Value is larger than the last threshold
+			if ( low <= value ) then return ThresholdTable.Colors[i] end
+
+		else
+			-- Value is in between 2 adjacent thresholds
+			if ( low <= value and value < high ) then
+				return ThresholdTable.Colors[i]
+			end
+		end
+	end
+
+	-- Should never reach here
+	return _G["GRAY_FONT_COLOR"];
+end
+
+--[[ API
+NAME: TitanUtils_ToString
+DESC: Routine that returns the text or an empty string.
+VAR: text - text to check
+OUT: string - string of text or ""
+--]]
+function TitanUtils_ToString(text)
+	return TitanUtils_Ternary(text, text, "");
+end
+
+--------------------------------------------------------------
+--
+-- Right click menu routines for plugins
+-- The expected global function name in the plugin is:
+-- "TitanPanelRightClickMenu_Prepare"..<registry.id>.."Menu"
+--
+--[[ API
+NAME: TitanPanelRightClickMenu_AddTitle
+DESC: Menu - add a title at the given level in the form of a button.
+VAR: title - text to show
+VAR: level - level to put text
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_AddTitle(title, level)
+	if (title) then
+		local info = {};
+		info.text = title;
+		info.notCheckable = true;
+		info.notClickable = true;
+		info.isTitle = 1;
+		L_UIDropDownMenu_AddButton(info, level);
+	end
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddCommand
+DESC: Menu - add a command at the given level in the form of a button.
+VAR: title - text to show
+VAR: value - value of the command
+VAR: functionName - routine to run when clicked
+VAR: level - level to put command
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_AddCommand(text, value, functionName, level)
+	local info = {};
+	info.notCheckable = true;
+	info.text = text;
+	info.value = value;
+	info.func = function()
+	local callback = _G[functionName];
+-- callback must be a function else do nothing (spank developer)
+		if callback and type(callback)== "function" then
+			callback(value)
+		end
+	end
+	L_UIDropDownMenu_AddButton(info, level);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddSpacer
+DESC: Menu - add a blank line at the given level in the form of an inactive button.
+VAR: level - level to put the line
+OUT: None
+--]]
+function TitanPanelRightClickMenu_AddSpacer(level)
+	local info = {};
+	info.notCheckable = true;
+	info.notClickable = true;
+	info.disabled = 1;
+	L_UIDropDownMenu_AddButton(info, level);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_Hide
+DESC: This will remove the plugin from the Titan bar.
+VAR: value - id of the plugin
+OUT: None
+--]]
+function TitanPanelRightClickMenu_Hide(value)
+	TitanPanel_RemoveButton(value);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddToggleVar
+DESC: Menu - add a toggle variable command at the given level in the form of a button.
+VAR: text - text to show
+VAR: id - id of the plugin
+VAR: var - the saved variable of the plugin to toggle
+VAR: toggleTable - control table (called with other than nil??)
+VAR: level - level to put the line
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_AddToggleVar(text, id, var, toggleTable, level)
+	local info = {};
+	info.text = text;
+	info.value = {id, var, toggleTable};
+	info.func = function()
+		TitanPanelRightClickMenu_ToggleVar({id, var, toggleTable})
+	end
+	info.checked = TitanGetVar(id, var);
+	info.keepShownOnClick = 1;
+	L_UIDropDownMenu_AddButton(info, level);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddToggleIcon
+DESC: Menu - add a toggle Icon (localized) command at the given level in the form of a button. Titan will properly control the "ShowIcon"
+VAR: id - id of the plugin
+VAR: level - level to put the line
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_AddToggleIcon(id, level)
+	TitanPanelRightClickMenu_AddToggleVar(L["TITAN_PANEL_MENU_SHOW_ICON"],
+	id, "ShowIcon", nil, level);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddToggleLabelText
+DESC: Menu - add a toggle Label (localized) command at the given level in the form of a button. Titan will properly control the "ShowLabelText"
+VAR: id - id of the plugin
+VAR: level - level to put the line
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_AddToggleLabelText(id, level)
+	TitanPanelRightClickMenu_AddToggleVar(L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"],
+	id, "ShowLabelText", nil, level);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddToggleColoredText
+DESC: Menu - add a toggle Colored Text (localized) command at the given level in the form of a button. Titan will properly control the "ShowColoredText"
+VAR: id - id of the plugin
+VAR: level - level to put the line
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_AddToggleColoredText(id, level)
+	TitanPanelRightClickMenu_AddToggleVar(L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"],
+	id, "ShowColoredText", nil, level);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddHide
+DESC: Menu - add a Hide (localized) command at the given level in the form of a button. When clicked this will remove the plugin from the Titan bar.
+VAR: id - id of the plugin
+VAR: level - level to put the line
+OUT: None
+--]]
+function TitanPanelRightClickMenu_AddHide(id, level)
+	local info = {};
+	info.notCheckable = true;
+	info.text = L["TITAN_PANEL_MENU_HIDE"];
+	info.value = value;
+	info.func = function()
+		TitanPanelRightClickMenu_Hide(id)
+	end
+	L_UIDropDownMenu_AddButton(info, level);
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_ToggleVar
+DESC: This will toggle the Titan variable and the update the button.
+VAR: value - table of (id of the plugin, saved var to be updated, control table)
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_ToggleVar(value)
+	local id, var, toggleTable = nil, nil, nil;
+
+	-- table expected else do nothing
+	if type(value)~="table" then return end
+
+	if value and value[1] then id = value[1] end
+	if value and value[2] then var = value[2] end
+	if value and value[3] then toggleTable = value[3] end
+
+	-- Toggle var
+	TitanToggleVar(id, var);
+
+	if ( TitanPanelRightClickMenu_AllVarNil(id, toggleTable) ) then
+		-- Undo if all vars in toggle table nil
+		TitanToggleVar(id, var);
+	else
+		-- Otherwise continue and update the button
+		TitanPanelButton_UpdateButton(id, 1);
+	end
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AllVarNil
+DESC: Check if all the variables in the table are nil/false.
+VAR: id - id of the plugin
+VAR: toggleTable - table of saved var to be checked
+OUT: bool - true (1) or nil
+--]]
+function TitanPanelRightClickMenu_AllVarNil(id, toggleTable)
+	if ( toggleTable ) and type(toggleTable)== "table" then
+		for i, v in toggleTable do
+			if ( TitanGetVar(id, v) ) then
+				return nil;
+			end
+		end
+		return 1;
+	end
+end
+
+--[[ API
+NAME: TitanPanelRightClickMenu_AddToggleColoredText
+DESC: This will toggle the "ShowColoredText" Titan variable then update the button
+VAR: id - id of the plugin
+VAR: level - level to put the line
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_ToggleColoredText(value)
+	TitanToggleVar(value, "ShowColoredText");
+	TitanPanelButton_UpdateButton(value, 1);
+end
+
+--------------------------------------------------------------
+--
+-- Plugin manipulation routines
+--
+--[[ local
+NAME: TitanUtils_SwapButtonOnBar
+DESC: This will swap two buttons on the Titan bars. Once swapped then 'reinit' the buttons to show properly. This is currently used as part of the shift left / right.
+VAR: from_id - id of the plugin
+VAR: to_id - id of the plugin
+OUT:  None
+--]]
+local function TitanUtils_SwapButtonOnBar(from_id, to_id)
+	-- Used as part of the shift L / R to swap the buttons
+	local button = TitanPanelSettings.Buttons[from_id]
+	local locale = TitanPanelSettings.Location[from_id]
+
+	TitanPanelSettings.Buttons[from_id] = TitanPanelSettings.Buttons[to_id]
+	TitanPanelSettings.Location[from_id] = TitanPanelSettings.Location[to_id]
+	TitanPanelSettings.Buttons[to_id] = button
+	TitanPanelSettings.Location[to_id] = locale
+	TitanPanel_InitPanelButtons();
+end
+
+--[[ local
+NAME: TitanUtils_GetNextButtonOnBar
+DESC: Find the next button that is on the same bar and is on the same side.
+VAR: bar - The Titan bar to search
+VAR: id - id of the plugin to see if there is a plugin next to it
+VAR: side - right or left
+OUT: int - index of the next button or nil if none found
+NOTE:
+-- buttons on Left are placed L to R; buttons on Right are placed R to L. Next and prev depend on which side we need to check.
+:NOTE
+--]]
+local function TitanUtils_GetNextButtonOnBar(bar, id, side)
+	-- find the next button that is on the same bar and is on the same side
+	-- return nil if not found
+	local index = TitanUtils_GetCurrentIndex(TitanPanelSettings.Buttons, id);
+
+	for i, id in pairs(TitanPanelSettings.Buttons) do
+		if TitanUtils_GetWhichBar(id) == bar
+		and i > index
+		and TitanPanel_GetPluginSide(id) == side
+		and TitanUtils_IsPluginRegistered(id) then
+			return i;
+		end
+	end
+end
+
+--[[ local
+NAME: TitanUtils_GetPrevButtonOnBar
+DESC: Find the previous button that is on the same bar and is on the same side.
+VAR: bar - The Titan bar to search
+VAR: id - id of the plugin to see if there is a plugin previous to it
+VAR: side - right or left
+OUT: int - index of the previous button or nil if none found
+NOTE:
+-- buttons on Left are placed L to R; buttons on Right are placed R to L. Next and prev depend on which side we need to check.
+:NOTE
+--]]
+local function TitanUtils_GetPrevButtonOnBar(bar, id, side)
+	-- find the prev button that is on the same bar and is on the same side
+	-- return nil if not found
+	local index = TitanUtils_GetCurrentIndex(TitanPanelSettings.Buttons, id);
+	local prev_idx = nil
+
+	for i, id in pairs(TitanPanelSettings.Buttons) do
+		if TitanUtils_GetWhichBar(id) == bar
+		and i < index
+		and TitanPanel_GetPluginSide(id) == side
+		and TitanUtils_IsPluginRegistered(id) then
+			prev_idx = i; -- this might be the previous button
+		end
+		if i == index then
+			return prev_idx;
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanUtils_AddButtonOnBar
+DESC: Add the given plugin to the given bar. Then reinit the plugins to show it properly.
+VAR: bar - The Titan bar to add the plugin
+VAR: id - id of the plugin to add
+OUT:  None.
+--]]
+function TitanUtils_AddButtonOnBar(bar, id)
+	-- Add the button to the requested bar, if shown
+	if (not bar)
+	or (not id)
+	or (not TitanPanelSettings)
+	or (not TitanPanelGetVar(bar.."_Show"))
+	then
+		return;
+	end
+
+	local i = TitanPanel_GetButtonNumber(id)
+--[[
+TitanDebug("AddB: "..(id or "?").." "..(bar or "?").." "
+..(TitanPanelSettings and "T" or "F").." "..(TitanPanelGetVar(bar.."_Show") and "T" or "F").." "
+..(i or "?").." "
+)
+--]]
+	-- The _GetButtonNumber returns +1 if not found so it is 'safe' to
+	-- update / add to the Location
+	TitanPanelSettings.Buttons[i] = (id or "?")
+	TitanPanelSettings.Location[i] = (bar or "Bar")
+	TitanPanel_InitPanelButtons();
+end
+
+--[[ Titan
+NAME: TitanUtils_GetFirstButtonOnBar
+DESC: Find the first button that is on the given bar and is on the given side.
+VAR: bar - The Titan bar to search
+VAR: side - right or left
+OUT: int - index of the first button or nil if none found
+NOTE:
+-- buttons on Left are placed L to R; buttons on Right are placed R to L. Next and prev depend on which side we need to check.
+-- buttons on Right are placed R to L
+:NOTE
+--]]
+function TitanUtils_GetFirstButtonOnBar(bar, side)
+	-- find the first button that is on the same bar and is on the same side
+	-- return nil if not found
+	local index = 0
+
+	for i, id in pairs(TitanPanelSettings.Buttons) do
+		if TitanUtils_GetWhichBar(id) == bar
+		and i > index
+		and TitanPanel_GetPluginSide(id) == side
+		and TitanUtils_IsPluginRegistered(id) then
+			return i;
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanUtils_ShiftButtonOnBarLeft
+DESC: Find the button that is on the bar and is on the side and left of the given button
+VAR:
+- name - id of the plugin
+OUT:  None
+--]]
+function TitanUtils_ShiftButtonOnBarLeft(name)
+	-- Find the button to the left. If there is one, swap it in the array
+	local from_idx = TitanUtils_GetCurrentIndex(TitanPanelSettings.Buttons,name)
+	local side = TitanPanel_GetPluginSide(name)
+	local bar = TitanUtils_GetWhichBar(name)
+
+	-- buttons on Left are placed L to R;
+	-- buttons on Right are placed R to L
+	if side and side == TITAN_LEFT then
+		to_idx = TitanUtils_GetPrevButtonOnBar (TitanUtils_GetWhichBar(name), name, side)
+	elseif side and side == TITAN_RIGHT then
+		to_idx = TitanUtils_GetNextButtonOnBar (TitanUtils_GetWhichBar(name), name, side)
+	end
+
+	if to_idx then
+		TitanUtils_SwapButtonOnBar(from_idx, to_idx);
+	else
+		return
+	end
+end
+
+--[[ Titan
+NAME: TitanUtils_ShiftButtonOnBarRight
+DESC: Find the button that is on the bar and is on the side and right of the given button
+VAR:
+- name - id of the plugin
+OUT:  None
+--]]
+function TitanUtils_ShiftButtonOnBarRight(name)
+	-- Find the button to the right. If there is one, swap it in the array
+	local from_idx = TitanUtils_GetCurrentIndex(TitanPanelSettings.Buttons,name)
+	local to_idx = nil
+	local side = TitanPanel_GetPluginSide(name)
+	local bar = TitanUtils_GetWhichBar(name)
+
+	-- buttons on Left are placed L to R;
+	-- buttons on Right are placed R to L
+	if side and side == TITAN_LEFT then
+		to_idx = TitanUtils_GetNextButtonOnBar (bar, name, side)
+	elseif side and side == TITAN_RIGHT then
+		to_idx = TitanUtils_GetPrevButtonOnBar (bar, name, side)
+	end
+
+	if to_idx then
+		TitanUtils_SwapButtonOnBar(from_idx, to_idx);
+	else
+		return
+	end
+end
+
+--------------------------------------------------------------
+--
+-- Frame check & manipulation routines
+--
+function TitanUtils_CheckFrameCounting(frame, elapsed)
+	if (frame:IsVisible()) then
+		if (not frame.frameTimer or not frame.isCounting) then
+			return;
+		elseif ( frame.frameTimer < 0 ) then
+			frame:Hide();
+			frame.frameTimer = nil;
+			frame.isCounting = nil;
+		else
+			frame.frameTimer = frame.frameTimer - elapsed;
+		end
+	end
+end
+
+function TitanUtils_StartFrameCounting(frame, frameShowTime)
+	frame.frameTimer = frameShowTime;
+	frame.isCounting = 1;
+end
+
+function TitanUtils_StopFrameCounting(frame)
+	frame.isCounting = nil;
+end
+
+function TitanUtils_CloseAllControlFrames()
+	for index, value in pairs(TitanPlugins) do
+		local frame = _G["TitanPanel"..index.."ControlFrame"];
+		if (frame and frame:IsVisible()) then
+			frame:Hide();
+		end
+	end
+end
+
+function TitanUtils_IsAnyControlFrameVisible() -- need?
+	for index, value in TitanPlugins do
+		local frame = _G["TitanPanel"..index.."ControlFrame"];
+		if (frame:IsVisible()) then
+			return true;
+		end
+	end
+	return false;
+end
+
+function TitanUtils_GetOffscreen(frame)
+	local offscreenX, offscreenY;
+	local ui_scale = UIParent:GetEffectiveScale()
+	if not frame then
+		return
+	end
+	local fr_scale = frame:GetEffectiveScale()
+
+	if ( frame and frame:GetLeft()
+	and frame:GetLeft() * fr_scale < UIParent:GetLeft() * ui_scale ) then
+		offscreenX = -1;
+	elseif ( frame and frame:GetRight()
+	and frame:GetRight() * fr_scale > UIParent:GetRight() * ui_scale ) then
+		offscreenX = 1;
+	else
+		offscreenX = 0;
+	end
+
+	if ( frame and frame:GetTop()
+	and frame:GetTop() * fr_scale > UIParent:GetTop() * ui_scale ) then
+		offscreenY = -1;
+	elseif ( frame and frame:GetBottom()
+	and frame:GetBottom() * fr_scale < UIParent:GetBottom() * ui_scale ) then
+		offscreenY = 1;
+	else
+		offscreenY = 0;
+	end
+
+	return offscreenX, offscreenY;
+end
+
+--------------------------------------------------------------
+--
+-- Plugin registration routines
+--
+--[[ Titan
+NAME: TitanUtils_PluginToRegister
+DESC: Place the plugin to be registered later by Titan
+VAR:
+- self - frame of the plugin (must be a Titan template)
+- isChildButton - true if the frame is a child of a Titan frame
+OUT:  None
+NOTE:
+- .registry is part of 'self' (the Titan plugin frame) which works great for Titan specific plugins.
+  Titan plugins create the registry as part of the frame _OnLoad.
+  But this does not work for LDB buttons. The frame is created THEN the registry is added to the frame.
+- Any read of the registry must assume it may not exist. Also assume the registry could be updated after this routine.
+- This is called when a Titan plugin frame is created. Normally these are held until the player 'enters world' then the plugin is registered.
+  Sometimes plugin frames are created after this process. Right now only LDB plugins are handled. If someone where to start creating Titan frames after the registration process were complete then it would fail to be registered...
+-!For LDB plugins the 'registry' is attached to the frame AFTER the frame is created...
+- The fields put into "Attempted" are defaulted here in preperation of being registered.
+--]]
+function TitanUtils_PluginToRegister(self, isChildButton)
+	TitanPluginToBeRegisteredNum = TitanPluginToBeRegisteredNum + 1
+	local cat = ""
+	local notes = ""
+	if self and self.registry then
+		cat = (self.registry.category or "")
+		notes = (self.registry.notes or "")
+	end
+	-- Some of the fields in this record are displayed in the "Attempts"
+	-- so they are defaulted here.
+	TitanPluginToBeRegistered[TitanPluginToBeRegisteredNum] = {
+		self = self,
+		button = ((self and self:GetName()
+			or "Nyl".."_"..TitanPluginToBeRegisteredNum)),
+		isChild = (isChildButton and true or false),
+		-- fields below are updated when registered
+		name = "?",
+		issue = "",
+		status = TITAN_NOT_REGISTERED,
+		category = cat,
+		plugin_type = "",
+		notes = notes,
+	}
+end
+
+--[[ Titan
+NAME: TitanUtils_PluginFail
+DESC: Place the plugin to be registered later by Titan
+VAR:
+- plugin - frame of the plugin (must be a Titan template)
+OUT:  None
+NOTE:
+- This is called when a plugin is unsupported. Cuurently this is used if a LDB data object is not supported. See SupportedDOTypes in LDBToTitan.lua for more detail.
+  It is intended mainly for developers. It is a place to put relevant info for debug and so users can supply troubleshooting info.
+  The key is set the status to 'fail' so there is no further attempt to register the plugin.
+- The results will show in "Attempted" so the developer has a shot at figuring out what was wrong.
+- plugin is expected to hold as much relevant info as possible...
+--]]
+function TitanUtils_PluginFail(plugin)
+	TitanPluginToBeRegisteredNum = TitanPluginToBeRegisteredNum + 1
+	TitanPluginToBeRegistered[TitanPluginToBeRegisteredNum] =
+		{
+		self = plugin.self,
+		button = (plugin.button and plugin.button:GetName() or ""),
+		isChild = (plugin.isChild and true or false),
+		name = (plugin.name or "?"),
+		issue = (plugin.issue or "?"),
+		status = TITAN_REGISTER_FAILED,
+		category = (plugin.category or ""),
+		plugin_type = (plugin.plugin_type or ""),
+		}
+end
+
+--[[ local
+NAME: TitanUtils_RegisterPluginProtected
+DESC: This routine is intended to be called in a protected manner (pcall) by Titan when it attempts to register a plugin.
+VAR:
+- plugin - frame of the plugin (must be a Titan template)
+OUT:
+- table
+	.issue	: Show the user what prevented the plugin from registering
+	.result	: Used so we know which plugins were processed
+	.id		: The name used to lookup the plugin
+	.cat		: The 'bucket' to use off the main Titan menu
+	.ptype	: For now just Titan or LDB type
+NOTE:
+- We try to anticipate the various ways a plugin could fail to register or just plain fail.
+  The intent is to keep Titan whole so a plugin does not prevent Titan from loading.
+  And attempt to tell the user / developer what went wrong.
+- If successful the plugin will be in TitanPlugins as a registered plugin and will be available for display on the Titan bars.
+--]]
+local function TitanUtils_RegisterPluginProtected(plugin)
+	local result = ""
+	local issue = ""
+	local id = ""
+	local cat = ""
+	local ptype = ""
+	local notes = ""
+
+	local self = plugin.self
+	local isChildButton = (plugin.isChild and true or false)
+
+	if self and self:GetName() then
+		if (isChildButton) then
+			-- This is a button within a button
+			self:RegisterForClicks("LeftButtonUp", "RightButtonUp", "MiddleButtonUp");
+			self:RegisterForDrag("LeftButton")
+			TitanPanelDetectPluginMethod(self:GetName(), true);
+			result = TITAN_REGISTERED
+			-- give some indication that this is valid...
+			id = (self:GetName() or "").."<child>"
+		else
+			-- Check for the .registry where all the Titan plugin info is expected
+			if (self.registry and self.registry.id) then
+				id = self.registry.id
+				if TitanUtils_IsPluginRegistered(id) then
+					-- We have already registered this plugin!
+					issue =  "Plugin already loaded. "
+					.."Please see if another plugin (Titan or LDB) is also loading "
+					.."with the same name.\n"
+					.."<Titan>.registry.id or <LDB>.label"
+				else
+					-- A sanity check just in case it was already in the list
+					if (not TitanUtils_TableContainsValue(TitanPluginsIndex, id)) then
+						-- Assign and Sort the list of plugins
+						TitanPlugins[id] = self.registry;
+						table.insert(TitanPluginsIndex, self.registry.id);
+						table.sort(TitanPluginsIndex,
+							function(a, b)
+								-- if the .menuText is missing then use .id
+								if TitanPlugins[a].menuText == nil then
+									TitanPlugins[a].menuText = TitanPlugins[a].id;
+								end
+								if TitanPlugins[b].menuText == nil then
+									TitanPlugins[b].menuText = TitanPlugins[b].id;
+								end
+								return string.lower(TitanPlugins[a].menuText)
+									< string.lower(TitanPlugins[b].menuText);
+							end
+						);
+					end
+				end
+				if issue ~= "" then
+					result = TITAN_REGISTER_FAILED
+				else
+					-- We are almost done-
+					-- Allow mouse clicks on the plugin
+					local pluginID = TitanUtils_GetButtonID(self:GetName());
+					local plugin_id = TitanUtils_GetPlugin(pluginID);
+					if (plugin_id) then
+						self:RegisterForClicks("LeftButtonUp", "RightButtonUp", "MiddleButtonUp");
+						self:RegisterForDrag("LeftButton")
+						if (plugin_id.id) then
+							TitanPanelDetectPluginMethod(plugin_id.id);
+						end
+					end
+					result = TITAN_REGISTERED
+					-- determine the plugin category
+					cat = (self.registry.category or nil)
+					ptype = TITAN_ID -- Assume it is created for Titan
+					if self.registry.ldb then
+						-- Override the type with the LDB type
+						ptype = "LDB: '"..self.registry.ldb.."'"
+					end
+				end
+				notes = (self.registry.notes or "")
+			else
+				-- There could be a couple reasons the .registry was not found
+				result = TITAN_REGISTER_FAILED
+				if (not self.registry) then
+					issue = "Can not find registry for plugin (self.registry)"
+				end
+				if (self.registry and not self.registry.id) then
+					issue = "Can not determine plugin name (self.registry.id)"
+				end
+			end
+		end
+	else
+		-- The button could not be determined - the plugin is hopeless
+		result = TITAN_REGISTER_FAILED
+		issue = "Can not determine plugin button name"
+	end
+
+	-- create and return the results
+	local ret_val = {}
+	ret_val.issue = (issue or "")
+	ret_val.result = (result or TITAN_REGISTER_FAILED)
+	ret_val.id = (id or "")
+	ret_val.cat = (cat or "General")
+	ret_val.ptype = ptype
+	ret_val.notes = notes
+	return ret_val
+end
+
+--[[ Titan
+NAME: TitanUtils_RegisterPlugin
+DESC: Attempt to register a plugin that has requested to be registered
+VAR:
+- plugin - frame of the plugin (must be a Titan template)
+OUT:  None
+NOTE:
+- Lets be extremely paranoid here because registering plugins that do not play nice can cause real headaches...
+--]]
+function TitanUtils_RegisterPlugin(plugin)
+	local call_success, ret_val
+	-- Ensure we have a glimmer of a plugin and that the plugin has not
+	-- already been registered.
+	if plugin and plugin.status == TITAN_NOT_REGISTERED then
+		-- See if the request to register has a shot at success
+		if plugin.self then
+			-- Just in case, catch any errors
+			call_success, -- needed for pcall
+			ret_val =  -- actual return values
+				pcall (TitanUtils_RegisterPluginProtected, plugin)
+			-- pcall does not allow errors to propagate out. Any error
+			-- is returned as text with the success / fail.
+			-- Think of it as sort of a try - catch block
+			if call_success then
+				-- all is good so write the return values to the plugin
+				plugin.status = ret_val.result
+				plugin.issue = ret_val.issue
+				plugin.name = ret_val.id
+				plugin.category = ret_val.cat
+				plugin.notes = ret_val.notes
+				plugin.plugin_type = ret_val.ptype
+			else
+				-- write enough to the plugin so the user or developer
+				-- can see Titan at least tried...
+				plugin.status = TITAN_REGISTER_FAILED
+				plugin.issue = (ret_val.issue or "Unknown error")
+				plugin.name = "?"
+				plugin.notes = ret_val.notes or ""
+			end
+		else
+			-- write enough to the plugin so the user or developer can see something
+			plugin.status = TITAN_REGISTER_FAILED
+			plugin.issue = "Can not determine plugin button name"
+			plugin.name = "?"
+		end
+
+		-- If there was an error tell the user.
+		if not plugin.issue == ""
+		or plugin.status ~= TITAN_REGISTERED then
+			TitanDebug(TitanUtils_GetRedText("Error Registering Plugin")
+				..TitanUtils_GetGreenText(
+					": "
+					.."name: '"..(plugin.name or "?_").."' "
+					.."issue: '"..(plugin.issue or "?_").."' "
+					.."button: '"..plugin.button.."' "
+					)
+				)
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanUtils_RegisterPluginList
+DESC: Attempt to register the list of plugins that have requested to be registered
+VAR:  None
+OUT:  None
+NOTE:
+- Tell the user when this starts and ends only on the first time.
+  This could be called if a plugin requests to be registered after the first loop through.
+--]]
+function TitanUtils_RegisterPluginList()
+	-- Loop through the plugins that have requested to be loaded into Titan.
+	local result = ""
+	local issue = ""
+	local id
+	local cnt = 0
+	if TitanPluginToBeRegisteredNum > 0 then
+		if not Titan__InitializedPEW and not TitanAllGetVar("Silenced") then
+			TitanDebug(L["TITAN_PANEL_REGISTER_START"], "normal")
+		end
+		for index, value in ipairs(TitanPluginToBeRegistered) do
+			if TitanPluginToBeRegistered[index] then
+				TitanUtils_RegisterPlugin(TitanPluginToBeRegistered[index])
+			end
+			cnt = cnt + 1
+		end
+		if not Titan__InitializedPEW and not TitanAllGetVar("Silenced") then
+			TitanDebug((L["TITAN_PANEL_REGISTER_END"].." "..cnt), "normal")
+		end
+	end
+end
+
+--[[ API
+NAME: TitanUtils_IsPluginRegistered
+DESC: See if the given plugin was registered successfully.
+VAR:
+- id - id of the plugin
+OUT:  None
+- true (successful) or false
+--]]
+function TitanUtils_IsPluginRegistered(id)
+	if (id and TitanPlugins[id]) then
+		return true;
+	else
+		return false;
+	end
+end
+
+--------------------------------------------------------------
+-- Right click menu routines for Titan Panel bars
+
+--[[ Titan
+NAME: TitanUtils_CloseRightClickMenu
+DESC: Close the right click menu of any plugin if it was open. Only one can be open at a time.
+VAR:  None
+OUT:  None
+--]]
+function TitanUtils_CloseRightClickMenu()
+	if (DropDownList1:IsVisible()) then
+		DropDownList1:Hide();
+	end
+end
+
+--[[ local
+NAME: TitanRightClick_UIScale
+DESC: Scale the right click menu to the user requested value.
+VAR:  None
+OUT:
+- float - x scaled
+- float - y scaled
+- float - scale used
+
+--]]
+local function TitanRightClick_UIScale()
+	-- take UI Scale into consideration
+	local listFrame = _G["DropDownList1"];
+	local listframeScale = listFrame:GetScale();
+
+	local uiScale;
+	local uiParentScale = UIParent:GetScale();
+
+	local x, y = GetCursorPosition(UIParent)
+
+	if ( GetCVar("useUIScale") == "1" ) then
+		uiScale = tonumber(GetCVar("uiscale"));
+		if ( uiParentScale < uiScale ) then
+			uiScale = uiParentScale;
+		end
+	else
+		uiScale = uiParentScale;
+	end
+
+	x = x/uiScale;
+	y = y/uiScale;
+
+	listFrame:SetScale(uiScale);
+
+	return x, y, uiScale
+end
+
+--[[ local
+NAME: TitanRightClickMenu_OnLoad
+DESC: Prepare the plugin right click menu using the function given by the plugin.
+VAR:
+- plugin - frame of the plugin (must be a Titan template)
+OUT:  None
+NOTE:
+- The function name is assumed to be "TitanPanelRightClickMenu_Prepare"..plugin_id.."Menu".
+- This routine is for Titan plugins. There is a similar routine for the Titan bar.
+--]]
+local function TitanRightClickMenu_OnLoad(self)
+	local id = TitanUtils_GetButtonIDFromMenu(self);
+	if id then
+		local prepareFunction = _G["TitanPanelRightClickMenu_Prepare"..id.."Menu"]
+		if prepareFunction and type(prepareFunction) == "function" then
+			L_UIDropDownMenu_Initialize(self, prepareFunction, "MENU");
+		end
+	else
+		-- TitanDebug("Could not display tooltip. "
+		-- .."Could not determine Titan ID for "
+		-- .."'"..(self:GetName() or "?").."'. "
+		-- ,"error")
+	end
+end
+
+--[[ local
+NAME: TitanDisplayRightClickMenu_OnLoad
+DESC: Prepare the Titan bar right click menu using the given function.
+VAR:
+- self - frame of the Titan bar
+- func - function to create the menu
+OUT:  None
+NOTE:
+- This routine is for Titan bar. There is a similar routine for the Titan plugins.
+--]]
+local function TitanDisplayRightClickMenu_OnLoad(self, func)
+	local prepareFunction = _G[func];
+	if prepareFunction and type(prepareFunction) == "function" then
+		-- Nasty "hack", load Blizzard_Calendar if not loaded,
+		-- for it to secure init 24 dropdown menu buttons,
+		-- to avoid action blocked by tainting
+		if not IsAddOnLoaded("Blizzard_Calendar") then
+			LoadAddOn("Blizzard_Calendar")
+		end
+		-- not good practice but there seems to be no other way to get
+		-- the actual bar (frame parent) to the dropdown implementation
+		TitanPanel_DropMenu = self
+		L_UIDropDownMenu_Initialize(self, prepareFunction, "MENU");
+	end
+end
+
+--[[ local
+NAME: TitanPanelRightClickMenu_Toggle
+DESC: Call the routine to build the plugin menu then place it properly.
+VAR:
+- self - frame of the plugin (must be a Titan template)
+- isChildButton - function to create the menu
+OUT:  None
+NOTE:
+- This routine is for Titan plugins. There is a similar routine for the Titan bar.
+--]]
+function TitanPanelRightClickMenu_Toggle(self, isChildButton)
+	local x, y, scale
+	-- Get top / bottom
+	local name = self:GetName() -- assuming this is a plugin
+	local parent = self:GetParent():GetName()
+	local menu = _G[self:GetName().."RightClickMenu"]
+	local vert
+	local position
+	local id
+	local frame = ""
+
+	TitanRightClickMenu_OnLoad(menu)
+
+	-- if this is a child button then use the parent to get the plugin info
+	-- otherwise use self as passed in
+	if isChildButton then
+		id = TitanUtils_GetButtonID(parent)
+	else
+		id = TitanUtils_GetButtonID(name)
+	end
+	local i = TitanPanel_GetButtonNumber(id)
+	frame = TITAN_PANEL_DISPLAY_PREFIX..TitanPanelSettings.Location[i]
+	-- .Location would tell us the bar but we still need vert (top / bottom)
+	vert = TitanBarData[frame].vert
+	position = (vert == TITAN_TOP and TITAN_PANEL_PLACE_TOP or TITAN_PANEL_PLACE_BOTTOM)
+
+	if position == TITAN_PANEL_PLACE_TOP then
+		menu.point = "TOPLEFT";
+		menu.relativePoint = "BOTTOMLEFT";
+	else
+		menu.point = "BOTTOMLEFT";
+		menu.relativePoint = "TOPLEFT";
+	end
+
+	x, y, scale = TitanRightClick_UIScale()
+
+	L_ToggleDropDownMenu(1, nil, menu, frame, TitanUtils_Max(x - 40, 0), 0, nil, self);
+--[[ was not used...
+	local listFrame = _G["DropDownList"..L_UIDROPDOWNMENU_MENU_LEVEL];
+	local offscreenX, offscreenY = TitanUtils_GetOffscreen(listFrame);
+
+	if offscreenX == 1 then
+		if position == TITAN_PANEL_PLACE_TOP then
+			listFrame:ClearAllPoints();
+			listFrame:SetPoint("TOPRIGHT", frame, "BOTTOMLEFT", x, 0);
+		else
+			listFrame:ClearAllPoints();
+			listFrame:SetPoint("BOTTOMRIGHT", frame, "TOPLEFT", x, 0);
+		end
+	end
+--]]
+end
+
+--[[ Titan
+NAME: TitanPanelDisplayRightClickMenu_Toggle
+DESC: Call the routine to build the Titan bar menu then place it properly.
+VAR:
+- self - frame of the Titan bar
+- isChildButton - function to create the menu
+OUT:  None
+NOTE:
+- This routine is for Titan bar. There is a similar routine for the Titan plugins.
+- This is close to TitanPanelRightClickMenu_Toggle but geared to the Titan display bars. This routine allows the Titan display bars to be independent  rather than rely on bars being a 'sort of' plugin.
+- This relies on name="$parentRightClickMenu" being part of the display bar template.
+--]]
+function TitanPanelDisplayRightClickMenu_Toggle(self, isChildButton)
+	if not self:GetName() then
+		return
+	end
+
+	local frame = (isChildButton and self:GetParent():GetName() or self:GetName())
+	if not frame then
+		-- Only Titan display bars should be processed here!!!
+		return
+	end
+
+	local vert = TitanBarData[frame].vert
+	local position = (vert == TITAN_TOP and TITAN_PANEL_PLACE_TOP or TITAN_PANEL_PLACE_BOTTOM)
+	local x, y, scale
+	local menu
+
+	menu = _G[frame.."RightClickMenu"];
+	-- Initialize the DropDown Menu if not already initialized
+	TitanDisplayRightClickMenu_OnLoad(menu, "TitanPanelRightClickMenu_PrepareBarMenu")
+
+	if position == TITAN_PANEL_PLACE_TOP then
+		menu.point = "TOPLEFT";
+		menu.relativePoint = "BOTTOMLEFT";
+	else
+		menu.point = "BOTTOMLEFT";
+		menu.relativePoint = "TOPLEFT";
+	end
+
+	x, y, scale = TitanRightClick_UIScale()
+
+	L_ToggleDropDownMenu(1, nil, menu, frame, TitanUtils_Max(x - 40, 0), 0, nil, self)
+
+--[[ was not used...
+	local listFrame = _G["DropDownList"..L_UIDROPDOWNMENU_MENU_LEVEL];
+	local offscreenX, offscreenY = TitanUtils_GetOffscreen(listFrame);
+	if offscreenX == 1 then
+		if position == TITAN_PANEL_PLACE_TOP then
+			listFrame:ClearAllPoints();
+			listFrame:SetPoint("TOPRIGHT", frame, "BOTTOMLEFT", x, 0);
+		else
+			listFrame:ClearAllPoints();
+			listFrame:SetPoint("BOTTOMRIGHT", frame, "TOPLEFT", x, 0);
+		end
+	end
+--]]
+end
+
+--[[ Titan
+NAME: TitanPanelRightClickMenu_IsVisible
+DESC: Determine if a right click menu is shown. There can only be one.
+VAR:  None
+OUT:
+- true (IsVisible) or false
+--]]
+function TitanPanelRightClickMenu_IsVisible()
+	return _G["DropDownList1"]:IsVisible();
+end
+
+--[[ Titan
+NAME: TitanPanelRightClickMenu_Close
+DESC: Close the right click menu if shown. There can only be one.
+VAR:  None
+OUT:  None
+--]]
+function TitanPanelRightClickMenu_Close()
+	if _G["DropDownList1"]:IsVisible() then
+		_G["DropDownList1"]:Hide()
+	end
+end
+
+--------------------------------------------------------------
+-- Titan utility routines
+
+--[[ Titan
+NAME: TitanUtils_ParseName
+DESC: Parse the player name and return the parts.
+VAR:
+- name - the name to break up
+OUT:
+- string player name only
+- string realm name only
+--]]
+function TitanUtils_ParseName(name)
+	local server = ""
+	local player = ""
+	if name and name ~= TITAN_PROFILE_NONE then
+		local s, e, ident = string.find(name, TITAN_AT);
+		if s ~= nil then
+			server = string.sub(name, s+1);
+			player = string.sub(name, 1, s-1);
+		end
+	else
+	end
+	return player, server
+end
+
+--[[ Titan
+NAME: TitanUtils_CreateName
+DESC: Given the player name and server and return the Titan name.
+VAR:
+- player - 1st part
+- realm - 2nd part. Could be realm or 'custom'
+OUT:
+- string - Titan name
+--]]
+function TitanUtils_CreateName(player, realm)
+	local p1 = player or "?"
+	local p2 = realm or "?"
+
+	return p1..TITAN_AT..p2
+end
+
+--[[ Titan
+NAME: TitanUtils_GetPlayer
+DESC: Create the player name (toon being played) and return the parts.
+VAR:  None
+OUT:
+- string Titan player name or nil
+- string player name only
+- string realm name only
+--]]
+function TitanUtils_GetPlayer()
+	local playerName = UnitName("player");
+	local serverName = GetRealmName();
+	local toon = nil
+
+	if (playerName == nil
+	or serverName == nil
+--	Removed the following because people are using "Unknown" as a character name.
+--	or playerName == UNKNOWNOBJECT
+	or playerName == UKNOWNBEING) then
+		-- Do nothing if player name is not available
+	else
+		toon = playerName..TITAN_AT..serverName
+	end
+
+	return toon, playerName, serverName
+end
+
+--[[ Titan
+NAME: TitanUtils_GetGlobalProfile
+DESC: Return the global profile setting and the global profile name, if any.
+VAR:  None
+OUT:
+- bool Global profile value
+- string Global profile name or default
+- string player name only or blank
+- string realm name only or blank
+--]]
+function TitanUtils_GetGlobalProfile()
+	local playerName = ""
+	local serverName = ""
+	local glob = TitanAllGetVar("GlobalProfileUse")
+	local toon = TitanAllGetVar("GlobalProfileName")
+
+	if not toon then
+		-- this is a new install or toon
+		toon = TITAN_PROFILE_NONE
+		TitanAllSetVar("GlobalProfileName", TITAN_PROFILE_NONE)
+	end
+	if (toon == TITAN_PROFILE_NONE) then
+		--
+	else
+		-- If the profile name is not the default then split the name
+		playerName, serverName = TitanUtils_ParseName(toon)
+	end
+
+	return glob, toon, playerName, serverName
+end
+
+--[[ Titan
+NAME: TitanUtils_SetGlobalProfile
+DESC: Return the global profile setting and the global profile name, if any.
+VAR:
+- bool Global profile value
+- string Global profile name or default
+OUT:  None
+--]]
+function TitanUtils_SetGlobalProfile(glob, toon)
+	TitanAllSetVar("GlobalProfileUse", glob)
+	if glob then
+		-- The user asked for global
+		if toon == nil or toon == TITAN_PROFILE_NONE then
+			-- nothing was set before so use current player
+			toon = TitanUtils_GetPlayer()
+		end
+	end
+	TitanAllSetVar("GlobalProfileName", toon or TITAN_PROFILE_NONE)
+end
+
+--------------------------------------------------------------
+-- Various debug routines
+--[[
+local function Debug_array(message)
+local idx = TitanDebugArray.index
+	TitanDebugArray.index = mod(TitanDebugArray.index + 1, TITAN_DEBUG_ARRAY_MAX)
+	TitanDebugArray.lines[TitanDebugArray.index] = (date("%m/%d/%y %H:%M:%S".." : ")..message)
+end
+--]]
+--[[ Titan
+NAME: TitanPanel_GetVersion
+DESC: Get the Titan version into a string.
+VAR:  None
+OUT:
+- string containing the version
+--]]
+function TitanPanel_GetVersion()
+	return tostring(GetAddOnMetadata(TITAN_ID, "Version")) or L["TITAN_NA"];
+end
+--[[ Titan
+NAME: TitanPrint
+DESC: Output a message to the user in a consistent format.
+VAR:
+- message - string to output
+- msg_type - "info" | "warning" | "error" | "plain"
+OUT:
+- string - message to chat window
+--]]
+function TitanPrint(message, msg_type)
+	local dtype = ""
+	local pre = TitanUtils_GetGoldText(L["TITAN_PRINT"]..": ".._G["FONT_COLOR_CODE_CLOSE"])
+	local msg = ""
+	if msg_type == "error" then
+		dtype = TitanUtils_GetRedText("Error: ").._G["FONT_COLOR_CODE_CLOSE"]
+	elseif msg_type == "warning" then
+		dtype = "|cFFFFFF00".."Warning: ".._G["FONT_COLOR_CODE_CLOSE"]
+	elseif msg_type == "plain" then
+		pre = ""
+	elseif msg_type == "header" then
+		local ver = TitanPanel_GetVersion()
+		pre = TitanUtils_GetGoldText(L["TITAN_PANEL"])
+			..TitanUtils_GetGreenText(" "..ver)
+			..TitanUtils_GetGoldText(L["TITAN_PANEL_VERSION_INFO"]
+			)
+	end
+
+	msg = pre..dtype..TitanUtils_GetGreenText(message)
+	DEFAULT_CHAT_FRAME:AddMessage(msg)
+--	Debug_array(msg)
+end
+
+function TitanDebug(debug_message, debug_type)
+	local dtype = ""
+	local time_stamp = ""
+	local msg = ""
+	if debug_type == "error" then
+		dtype = TitanUtils_GetRedText("Error: ")
+	elseif debug_type == "warning" then
+		dtype = TitanUtils_GetHighlightText("Warning: ")
+	end
+	if debug_type == "normal" then
+		time_stamp = ""
+	else
+		time_stamp = TitanUtils_GetGoldText(date("%H:%M:%S")..": ")
+	end
+	if debug_message == true then
+		debug_message = "<true>";
+	end
+	if debug_message == false then
+		debug_message = "<false>";
+	end
+	if debug_message == nil then
+		debug_message = "<nil>";
+	end
+
+	msg =
+		TitanUtils_GetGoldText(L["TITAN_DEBUG"].." ")
+		..time_stamp
+		..dtype
+		..TitanUtils_GetGreenText(debug_message)
+
+	_G["DEFAULT_CHAT_FRAME"]:AddMessage(msg)
+--	Debug_array(msg)
+	--date("%m/%d/%y %H:%M:%S")
+end
+
+function TitanDumpPluginList()
+	-- Just dump the current list of plugins
+	for idx, value in pairs(TitanPluginsIndex) do
+		plug_in = TitanUtils_GetPlugin(TitanPluginsIndex[idx])
+		if plug_in then
+			TitanDebug("TitanDumpPluginList "
+				.."'"..idx.."'"
+				..": '"..(plug_in.id or "?").."'"
+				..": '"..(plug_in.version or "?").."'"
+			)
+		end
+	end
+end
+
+function TitanDumpPlayerList()
+	-- Just dump the current list of toons in Titan config
+	local cnt = 0
+	TitanDebug("TitanDumpPlayerList ==== start")
+	if TitanSettings.Players then
+		for idx, value in pairs(TitanSettings.Players) do
+			TitanDebug("-- "
+				.."'"..(idx or "?").."'"
+			)
+			cnt = cnt + 1
+		end
+	else
+		TitanDebug("No player list found!!! "
+			)
+	end
+	TitanDebug("TitanDumpPlayerList ==== done "..cnt)
+end
+
+function TitanDumpFrameName(self)
+	local frame
+	local parent
+	if self then
+		frame = self:GetName()
+	else
+		frame = "?"
+	end
+	if frame == "?" then
+		parent = "?"
+	else
+		parent = self:GetParent():GetName()
+	end
+--[
+TitanDebug("_GetFrameName "
+..(self and "T" or "F").." "
+..(frame or "?").." "
+..(parent or "?").." "
+)
+--]]
+end
+
+function TitanDumpTimers()
+	str = "Titan-timers: "
+		.."'"..(TitanAllGetVar("TimerPEW") or "?").."' "
+		.."'"..(TitanAllGetVar("TimerDualSpec") or "?").."' "
+		.."'"..(TitanAllGetVar("TimerLDB") or "?").."' "
+		.."'"..(TitanAllGetVar("TimerAdjust") or "?").."' "
+		.."'"..(TitanAllGetVar("TimerVehicle") or "?").."' "
+	TitanPrint(str, "plain")
+end
+
+function TitanArgConvert (event, a1, a2, a3, a4, a4, a5, a6)
+	local t1 = type(a1)
+	local t2 = type(a2)
+	local t3 = type(a3)
+	local t4 = type(a4)
+	local t5 = type(a5)
+	local t6 = type(a6)
+	if type(a1) == "boolean" then a1 = (a1 and "T" or "F") end
+	if type(a2) == "boolean" then a2 = (a2 and "T" or "F") end
+	if type(a3) == "boolean" then a3 = (a3 and "T" or "F") end
+	if type(a4) == "boolean" then a4 = (a4 and "T" or "F") end
+	if type(a5) == "boolean" then a5 = (a5 and "T" or "F") end
+	if type(a6) == "boolean" then a6 = (a6 and "T" or "F") end
+	TitanDebug(event.." "
+		.."1: "..(a1 or "?").."("..t1..") "
+		.."2: "..(a2 or "?").."("..t2..") "
+		.."3: "..(a3 or "?").."("..t3..") "
+		.."4: "..(a4 or "?").."("..t4..") "
+		.."5: "..(a5 or "?").."("..t5..") "
+		.."6: "..(a6 or "?").."("..t6..") "
+	)
+end
+
+--------------------------------------------------------------
+--
+-- Deprecated routines
+-- These routines will be commented out for a couple releases then deleted.
+--
+--[[
+
+--]]
diff --git a/Titan/TitanVariables.lua b/Titan/TitanVariables.lua
new file mode 100644
index 0000000..7d507bc
--- /dev/null
+++ b/Titan/TitanVariables.lua
@@ -0,0 +1,1063 @@
+--[[ File
+NAME: TitanVariables.lua
+DESC: This file contains the routines to initialize, get, and set the basic data structures used by Titan.
+It also sets the global variables and constants used by Titan.
+
+TitanSettings, TitanSkins, ServerTimeOffsets, ServerHourFormat are the structures saved to disk (listed in toc).
+TitanSettings: is the table that holds the Titan variables by character and the plugins used by that character.
+TitanSkins: is the table that holds the list of Titan and custom skins available to the user. It is assumed that the skins are in the proper folder on the hard drive. Blizzard does not allow addons to access the disk.
+ServerTimeOffsets and ServerHourFormat: are the tables that hold the user selected hour offset and display format per realm (server).
+:DESC
+--]]
+local L = LibStub("AceLocale-3.0"):GetLocale(TITAN_ID, true)
+local _G = getfenv(0);
+local media = LibStub("LibSharedMedia-3.0")
+
+-- Global variables
+Titan__InitializedPEW = nil
+Titan__Initialized_Settings = nil
+
+TITAN_AT = "@"
+
+TitanAll = nil;
+TitanSettings = nil;
+TitanPlayerSettings = nil
+TitanPluginSettings = nil;  -- Used by plugins
+TitanPanelSettings = nil;
+
+TITAN_PANEL_UPDATE_BUTTON = 1;
+TITAN_PANEL_UPDATE_TOOLTIP = 2;
+TITAN_PANEL_UPDATE_ALL = 3;
+TitanTooltipOrigScale = 1;
+TitanTooltipScaleSet = 0;
+
+-- Set Titan Version var for backwards compatibility
+TITAN_VERSION = GetAddOnMetadata(TITAN_ID, "Version") or L["TITAN_NA"]
+
+-- Various constants
+TITAN_PANEL_PLACE_TOP = 1;
+TITAN_PANEL_PLACE_BOTTOM = 2;
+TITAN_PANEL_PLACE_BOTH = 3;
+TITAN_PANEL_MOVING = 0;
+
+TITAN_WOW_SCREEN_TOP = 768
+TITAN_WOW_SCREEN_BOT = 0
+
+TITAN_TOP = "Top"
+TITAN_BOTTOM = "Bottom"
+TITAN_RIGHT = "Right"
+TITAN_LEFT = "Left"
+TITAN_PANEL_BUTTONS_ALIGN_LEFT = 1;
+TITAN_PANEL_BUTTONS_ALIGN_CENTER = 2;
+
+TITAN_PANEL_CONTROL = "TitanPanelBarButton"
+-- New bar vars
+TITAN_PANEL_BAR_HEIGHT = 24
+TITAN_PANEL_BAR_TEXTURE_HEIGHT = 30
+TITAN_PANEL_AUTOHIDE_PREFIX = "TitanPanelAutoHide_"
+TITAN_PANEL_AUTOHIDE_SUFFIX = "Button"
+TITAN_PANEL_HIDE_PREFIX = "Titan_Bar__Hider_"
+TITAN_PANEL_DISPLAY_PREFIX = "Titan_Bar__Display_"
+TITAN_PANEL_DISPLAY_MENU = "Menu_"
+--TITAN_PANEL_DISPLAY_ID = "Id_"
+TITAN_PANEL_BACKGROUND_PREFIX = "TitanPanelBackground_"
+TITAN_PANEL_TEXT = "Text"
+TITAN_PANEL_BUTTON_TEXT = "Button"..TITAN_PANEL_TEXT
+TITAN_PANEL_CONSTANTS = {
+	FONT_SIZE = 10,
+	FONT_NAME = "Friz Quadrata TT"
+}
+if (GetLocale() == "ruRU") then
+	-- Special fix for Russian - "Friz Quadrata TT" does not seem to work
+	TITAN_PANEL_CONSTANTS.FONT_NAME = "Arial Narrow"
+end
+local TPC = TITAN_PANEL_CONSTANTS -- shortcut
+
+TITAN_CUSTOM_PROFILE_POSTFIX = "TitanCustomProfile"
+TITAN_PROFILE_NONE = "<>"
+TITAN_PROFILE_RESET = "<RESET>"
+TITAN_PROFILE_USE = "<USE>"
+TITAN_PROFILE_INIT = "<INIT>"
+
+AUTOHIDE_PREFIX = "TitanPanelAutoHide_"
+AUTOHIDE_SUFFIX = "Button"
+
+TITAN_PANEL_BUTTONS_PLUGIN_CATEGORY =
+	{"Built-ins","General","Combat","Information","Interface","Profession"}
+
+--[[ Titan
+NAME: Titan bar overview
+DESC:
+-- 3 buttons are used to create a Titan bar:
+-- the 'display' button,
+-- the 'hider',
+-- and the 'auto hide' plugin.
+-- The display is where the plugins are displayed.
+-- The hider is used if auto hide is requested. This button will cause the display to show when the mouse is enters.
+-- The auto hide is the plugin that shows the auto hide 'pin'.
+:DESC
+--]]
+--[[ Titan
+NAME: TitanBarOrder table
+DESC:
+The values must match the 'name' in the TitanBarData table!!!
+The values specify the order the options should be ordered in the options pulldown.
+:DESC
+--]]
+TitanBarOrder = {"Bar", "Bar2", "AuxBar2", "AuxBar"}
+--[[ Titan
+NAME: TitanBarData table.
+DESC:
+The index must match the 'button' names in the TitanPanel.xml!!!
+The table holds:
+ the name of each Titan bar (in the index)
+ the short name of the bar
+ whether the bar is on top or bottom
+ the order they should be in top to bottom
+ show / hide give the values for the cooresponding SetPoint
+
+The short name is used to build names of the various saved variables, frames,
+ and buttons used by Titan.
+:DESC
+-]]
+TitanBarData = {
+	[TITAN_PANEL_DISPLAY_PREFIX.."Bar"] = {
+		locale_name = L["TITAN_PANEL_MENU_TOP"],
+		name = "Bar", vert = TITAN_TOP, order = 1,
+		hider = TITAN_PANEL_HIDE_PREFIX.."Bar",
+		auto_hide_plugin = AUTOHIDE_PREFIX.."Bar"..AUTOHIDE_SUFFIX,
+		plugin_y_offset = 1,
+		show = {
+		top = {pt="TOPLEFT", rel_fr="UIParent", rel_pt="TOPLEFT", x=0, y=0},
+		bot = {pt="BOTTOMRIGHT", rel_fr="UIParent", rel_pt="TOPRIGHT", x=0, y=-TITAN_PANEL_BAR_HEIGHT} },
+		hide = {
+		top = {pt="TOPLEFT", rel_fr="UIParent", rel_pt="TOPLEFT", x=0, y=TITAN_PANEL_BAR_HEIGHT*2},
+		bot = {pt="BOTTOMRIGHT", rel_fr="UIParent", rel_pt="TOPRIGHT", x=0, y=TITAN_PANEL_BAR_HEIGHT*2} }
+	},
+	[TITAN_PANEL_DISPLAY_PREFIX.."Bar2"] = {
+		locale_name = L["TITAN_PANEL_MENU_TOP2"],
+		name = "Bar2", vert = TITAN_TOP, order = 2,
+		hider = TITAN_PANEL_HIDE_PREFIX.."Bar2",
+		auto_hide_plugin = AUTOHIDE_PREFIX.."Bar2"..AUTOHIDE_SUFFIX,
+		plugin_y_offset = 1,
+		show = {
+		top = {pt="TOPLEFT", rel_fr="UIParent", rel_pt="TOPLEFT", x=0, y=-TITAN_PANEL_BAR_HEIGHT},
+		bot = {pt="BOTTOMRIGHT", rel_fr="UIParent", rel_pt="TOPRIGHT", x=0, y=-TITAN_PANEL_BAR_HEIGHT*2} },
+		hide = {
+		top = {pt="TOPLEFT", rel_fr="UIParent", rel_pt="TOPLEFT", x=0, y=TITAN_PANEL_BAR_HEIGHT*2},
+		bot = {pt="BOTTOMRIGHT", rel_fr="UIParent", rel_pt="TOPRIGHT", x=0, y=TITAN_PANEL_BAR_HEIGHT*2} }
+	},
+	-- no idea why -1 is needed for the bottom... seems anchoring to bottom is off a pixel
+	[TITAN_PANEL_DISPLAY_PREFIX.."AuxBar2"] = {
+		locale_name = L["TITAN_PANEL_MENU_BOTTOM2"],
+		name = "AuxBar2",  vert = TITAN_BOTTOM, order = 3,
+		hider = TITAN_PANEL_HIDE_PREFIX.."AuxBar2",
+		auto_hide_plugin = AUTOHIDE_PREFIX.."AuxBar2"..AUTOHIDE_SUFFIX,
+		plugin_y_offset = 1,
+		show = {
+		top = {pt="TOPRIGHT", rel_fr="UIParent", rel_pt="BOTTOMRIGHT", x=0, y=TITAN_PANEL_BAR_HEIGHT*2-1},
+		bot = {pt="BOTTOMLEFT", rel_fr="UIParent", rel_pt="BOTTOMLEFT", x=0, y=TITAN_PANEL_BAR_HEIGHT-1} },
+		hide = {
+		top = {pt="TOPRIGHT", rel_fr="UIParent", rel_pt="BOTTOMRIGHT", x=0, y=-TITAN_PANEL_BAR_HEIGHT*2},
+		bot = {pt="BOTTOMLEFT", rel_fr="UIParent", rel_pt="BOTTOMLEFT", x=0, y=-TITAN_PANEL_BAR_HEIGHT*2} }
+	},
+	[TITAN_PANEL_DISPLAY_PREFIX.."AuxBar"] = {
+		locale_name = L["TITAN_PANEL_MENU_BOTTOM"],
+		name = "AuxBar",  vert = TITAN_BOTTOM, order = 4,
+		hider = TITAN_PANEL_HIDE_PREFIX.."AuxBar",
+		auto_hide_plugin = AUTOHIDE_PREFIX.."AuxBar"..AUTOHIDE_SUFFIX,
+		plugin_y_offset = 1,
+		show = {
+		top = {pt="TOPRIGHT", rel_fr="UIParent", rel_pt="BOTTOMRIGHT", x=0, y=TITAN_PANEL_BAR_HEIGHT-1},
+		bot = {pt="BOTTOMLEFT", rel_fr="UIParent", rel_pt="BOTTOMLEFT", x=0, y=0-1} },
+		hide = {
+		top = {pt="TOPRIGHT", rel_fr="UIParent", rel_pt="BOTTOMRIGHT", x=0, y=-TITAN_PANEL_BAR_HEIGHT*2},
+		bot = {pt="BOTTOMLEFT", rel_fr="UIParent", rel_pt="BOTTOMLEFT", x=0, y=-TITAN_PANEL_BAR_HEIGHT*2} }
+	},
+	}
+
+-- Timers used within Titan
+TitanTimers = {}
+--[[ Titan
+NAME: AutoHideData table.
+DESC:
+The index must match the hider 'button' names in the TitanPanel.xml!!!
+The table holds:
+ the name of each hider bar (in the index)
+ the short name of the hider bar
+ whether the hider bar is on top or bottom
+:DESC
+--]]
+AutoHideData = { -- This has to follow the convention for plugins...
+	[AUTOHIDE_PREFIX.."Bar"..AUTOHIDE_SUFFIX] = {name = "Bar", vert = TITAN_TOP},
+	[AUTOHIDE_PREFIX.."Bar2"..AUTOHIDE_SUFFIX] = {name = "Bar2", vert = TITAN_TOP},
+	[AUTOHIDE_PREFIX.."AuxBar2"..AUTOHIDE_SUFFIX] = {name = "AuxBar2", vert = TITAN_BOTTOM},
+	[AUTOHIDE_PREFIX.."AuxBar"..AUTOHIDE_SUFFIX] = {name = "AuxBar", vert = TITAN_BOTTOM},
+	}
+
+--[[ Titan
+TitanPluginToBeRegistered table holds each plugin that is requesting to be a plugin.
+TitanPluginToBeRegisteredNum is the number of plugins that have requested.
+Each plugin in the table will be updated with the status of the registration and will be available in the Titan Attempted option.
+--]]
+TitanPluginToBeRegistered = {}
+TitanPluginToBeRegisteredNum = 0
+
+TitanPluginRegisteredNum = 0
+
+--TitanPluginAttempted = {}
+--TitanPluginAttemptedNum = 0
+
+--[[ Titan
+TitanPluginExtras table holds the plugin data for plugins that are in saved variables but not loaded on the current character.
+TitanPluginExtrasNum is the number of plugins not loaded.
+--]]
+TitanPluginExtras = {}
+TitanPluginExtrasNum = 0
+
+-- Global to hold where the Titan menu orginated from...
+TitanPanel_DropMenu = nil
+
+local Default_Plugins = {
+	{id = "Location", loc = "Bar"},
+	{id = "XP", loc = "Bar"},
+	{id = "Gold", loc = "Bar"},
+	{id = "Clock", loc = "Bar"},
+	{id = "Volume", loc = "Bar"},
+	{id = "AutoHide_Bar", loc = "Bar"},
+	{id = "Bag", loc = "Bar"},
+	{id = "Repair", loc = "Bar"},
+}
+--[[ Titan
+TITAN_PANEL_SAVED_VARIABLES table holds the Titan Panel Default SavedVars.
+--]]
+TITAN_PANEL_SAVED_VARIABLES = {
+	Buttons = {},
+	Location = {},
+	TexturePath = "Interface\\AddOns\\Titan\\Artwork\\",
+	Transparency = 0.7,
+	AuxTransparency = 0.7,
+	Scale = 1,
+	ButtonSpacing = 20,
+	IconSpacing = 0,
+	TooltipTrans = 1,
+	TooltipFont = 1,
+	DisableTooltipFont = 1,
+	FontName = TPC.FONT_NAME,
+	FrameStrata = "LOW",
+	FontSize = TPC.FONT_SIZE,
+	ScreenAdjust = false,
+	LogAdjust = false,
+	MinimapAdjust = false,
+	BagAdjust = 1,
+	TicketAdjust = 1,
+	Position = 1,
+	ButtonAlign = 1,
+	AuxScreenAdjust = false,
+	LockButtons = false,
+	LockAutoHideInCombat = false,
+	VersionShown = 1,
+	ToolTipsShown = 1,
+	HideTipsInCombat = false,
+	-- for the independent bars
+	Bar_Show = true,
+	Bar_Hide = false,
+	Bar_Align = TITAN_PANEL_BUTTONS_ALIGN_LEFT,
+	Bar_Transparency = 0.7,
+	Bar2_Show = false,
+	Bar2_Hide = false,
+	Bar2_Transparency = 0.7,
+	Bar2_Align = TITAN_PANEL_BUTTONS_ALIGN_LEFT,
+	AuxBar_Show = false,
+	AuxBar_Hide = false,
+	AuxBar_Transparency = 0.7,
+	AuxBar_Align = TITAN_PANEL_BUTTONS_ALIGN_LEFT,
+	AuxBar2_Show = false,
+	AuxBar2_Hide = false,
+	AuxBar2_Transparency = 0.7,
+	AuxBar2_Align = TITAN_PANEL_BUTTONS_ALIGN_LEFT,
+};
+
+--[[ Titan
+TITAN_ALL_SAVED_VARIABLES table holds the Titan Panel Global SavedVars.
+--]]
+TITAN_ALL_SAVED_VARIABLES = {
+	-- for timers in seconds
+	TimerPEW = 4,
+	TimerDualSpec = 2,
+	TimerLDB = 2,
+	TimerAdjust = 1,
+	TimerVehicle = 1,
+	-- Global profile
+	GlobalProfileUse = false,
+	GlobalProfileName = TITAN_PROFILE_NONE,
+	-- Silent Load
+	Silenced = false,
+	-- OrderHallCommandBar Status
+	OrderHall = true,
+};
+
+-- The skins released with Titan
+TitanSkinsDefaultPath = "Interface\\AddOns\\Titan\\Artwork\\"
+TitanSkinsCustomPath = TitanSkinsDefaultPath.."Custom\\"
+TitanSkinsPathEnd = "\\"
+TitanSkinsDefault = {
+	{ name = "Titan Default", titan=true, path = TitanSkinsDefaultPath},
+	{ name = "BlackPlusOne", titan=true, path = TitanSkinsCustomPath.."BlackPlusOne Skin"..TitanSkinsPathEnd},
+	{ name = "Christmas", titan=true, path = TitanSkinsCustomPath.."Christmas Skin"..TitanSkinsPathEnd},
+	{ name = "Charcoal Metal", titan=true, path = TitanSkinsCustomPath.."Charcoal Metal"..TitanSkinsPathEnd},
+	{ name = "Crusader", titan=true, path = TitanSkinsCustomPath.."Crusader Skin"..TitanSkinsPathEnd},
+	{ name = "Cursed Orange", titan=true, path = TitanSkinsCustomPath.."Cursed Orange Skin"..TitanSkinsPathEnd},
+	{ name = "Dark Wood", titan=true, path = TitanSkinsCustomPath.."Dark Wood Skin"..TitanSkinsPathEnd},
+	{ name = "Deep Cave", titan=true, path = TitanSkinsCustomPath.."Deep Cave Skin"..TitanSkinsPathEnd},
+	{ name = "Elfwood", titan=true, path = TitanSkinsCustomPath.."Elfwood Skin"..TitanSkinsPathEnd},
+	{ name = "Engineer", titan=true, path = TitanSkinsCustomPath.."Engineer Skin"..TitanSkinsPathEnd},
+	{ name = "Frozen Metal", titan=true, path = TitanSkinsCustomPath.."Frozen Metal Skin"..TitanSkinsPathEnd},
+	{ name = "Graphic", titan=true, path = TitanSkinsCustomPath.."Graphic Skin"..TitanSkinsPathEnd},
+	{ name = "Graveyard", titan=true, path = TitanSkinsCustomPath.."Graveyard Skin"..TitanSkinsPathEnd},
+	{ name = "Hidden Leaf", titan=true, path = TitanSkinsCustomPath.."Hidden Leaf Skin"..TitanSkinsPathEnd},
+	{ name = "Holy Warrior", titan=true, path = TitanSkinsCustomPath.."Holy Warrior Skin"..TitanSkinsPathEnd},
+	{ name = "Nightlife", titan=true, path = TitanSkinsCustomPath.."Nightlife Skin"..TitanSkinsPathEnd},
+	{ name = "Orgrimmar", titan=true, path = TitanSkinsCustomPath.."Orgrimmar Skin"..TitanSkinsPathEnd},
+	{ name = "Plate", titan=true, path = TitanSkinsCustomPath.."Plate Skin"..TitanSkinsPathEnd},
+	{ name = "Tribal", titan=true, path = TitanSkinsCustomPath.."Tribal Skin"..TitanSkinsPathEnd},
+	{ name = "X-Perl", titan=true, path = TitanSkinsCustomPath.."X-Perl"..TitanSkinsPathEnd},
+};
+TitanSkins = {}
+
+-- trim version if it exists
+local fullversion = GetAddOnMetadata(TITAN_ID, "Version")
+if fullversion then
+	local pos = string.find(fullversion, " -", 1, true);
+	if pos then
+		TITAN_VERSION = string.sub(fullversion,1,pos-1);
+	end
+end
+
+--[[ local
+NAME: TitanRegisterExtra
+DESC: Add the saved variable data of an unloaded plugin to the 'extra' list in case the user wants to delete the data via Tian Extras option.
+VAR: id - the name of the plugin (string)
+OUT:  None
+--]]
+local function TitanRegisterExtra(id)
+	TitanPluginExtrasNum = TitanPluginExtrasNum + 1
+	TitanPluginExtras[TitanPluginExtrasNum] =
+		{num=TitanPluginExtrasNum,
+		id = (id or "?"),
+		}
+end
+
+-- routines to sync toon data
+local function CleanupProfile ()
+	if TitanPanelSettings and TitanPanelSettings["Buttons"] then
+		-- Hide the current set of plugins to prevent overlap (creates a very messy bar!)
+		for index, id in pairs(TitanPanelSettings["Buttons"]) do
+			local currentButton =
+				TitanUtils_GetButton(TitanPanelSettings["Buttons"][index]);
+			-- safeguard
+			if currentButton then
+			currentButton:Hide();
+			end
+		end
+	end
+	TitanPanelRightClickMenu_Close();
+end
+
+--[[ local
+NAME: TitanVariables_SyncRegisterSavedVariables
+DESC: Helper routine to sync two sets of toon data - Titan settings and loaded plugins.
+VAR: registeredVariables - current loaded data (destination)
+VAR: savedVariables - data to compare with (source)
+OUT:  None
+--]]
+local function TitanVariables_SyncRegisterSavedVariables(registeredVariables, savedVariables)
+	if (registeredVariables and savedVariables) then
+		-- Init registeredVariables
+		for index, value in pairs(registeredVariables) do
+			if (not TitanUtils_TableContainsIndex(savedVariables, index)) then
+				savedVariables[index] = value;
+			end
+		end
+
+		-- Remove out-of-date savedVariables
+		for index, value in pairs(savedVariables) do
+			if (not TitanUtils_TableContainsIndex(registeredVariables, index)) then
+				savedVariables[index] = nil;
+			end
+		end
+	end
+end
+
+--[[ local
+NAME: Plugin_settings
+DESC: Give the curent profile the default plugins - if they are registered.
+VAR: reset - boolean
+OUT: None
+NOTE:
+- It is assumed this is a plugin wipe of the given profile.
+- Use the default Titan plugin list to display on the given bar.
+- These will be saved on exit or reload in the given profile.
+:NOTE
+--]]
+local function Plugin_settings(reset)
+	local plugin_list = {}
+	if reset then -- use the default install list
+		plugin_list = Default_Plugins
+	else -- use the current profile
+		plugin_list = TitanPanelSettings.Buttons
+	end
+
+	-- Init each and every default plugin
+	for idx, default_plugin in pairs(plugin_list) do
+		local id = default_plugin.id
+		local loc = default_plugin.loc
+		local plugin = TitanUtils_GetPlugin(id)
+--TitanDebug("Plugin: "..(id or "?").." "..(plugin and "T" or "F"))
+		-- See if plugin is registered
+		if (plugin) then
+--TitanDebug("__Plugin: "..(id or "?").." "..(loc or "?"))
+			-- Synchronize registered and saved variables
+			TitanVariables_SyncRegisterSavedVariables(
+				plugin.savedVariables, TitanPluginSettings[id])
+			TitanUtils_AddButtonOnBar(loc, id)
+			TitanPanelButton_UpdateButton(id)
+		end
+	end
+end
+
+--[[ local
+NAME: TitanVariables_PluginSettingsInit
+DESC: Give the curent profile the default plugins - if they are registered.
+VAR: None
+OUT: None
+NOTE:
+- The saved variables of the given profile will be used.
+- These will be saved on exit or reload in the given profile.
+- The saved display list will be used but only the registered plugins will be displayed.
+- The plugins that are not registered will NOT be removed from the saved list. This allows different a single saved display list to be used for toons that have different plugins enabled.
+:NOTE
+--]]
+local function TitanVariables_PluginSettingsInit()
+	-- Loop through the user's displayed plugins and see what is
+	-- actually registered
+	for idx, display_plugin in pairs(TitanPanelSettings.Buttons) do
+		local id = display_plugin
+		local plugin = TitanUtils_GetPlugin(id)
+		-- See if plugin is registered
+		if (plugin) then
+			-- Synchronize registered and saved variables
+			TitanVariables_SyncRegisterSavedVariables(
+				plugin.savedVariables, TitanPluginSettings[id])
+			-- Button will be updated later
+		else
+			-- Do not display this plugin.
+			-- Do NOT remove the button from the displayed list.
+			-- This is an old 'feature' that people like...
+		end
+	end
+end
+
+--[[ local
+NAME: TitanVariables_SyncSkins
+DESC: Routine to sync two sets of skins data - Titan defaults and Titan saved vars.
+VAR:  None
+OUT:  None
+NOTE:
+- It is assumed that the list in Titan defaults or as input from the user are in the Titan skins folder. Blizz does not allow LUA to read the hard drive directly.
+:NOTE
+--]]
+local function TitanVariables_SyncSkins()
+	if (TitanSkinsDefault and TitanSkins) then
+		local skins = {}
+		-- insert all the Titan defaults
+		for idx, v in pairs(TitanSkinsDefault) do
+			table.insert (skins, TitanSkinsDefault[idx])
+--			table.sort(skins, function(a, b)
+--				return string.lower(skins[a] and skins[a].name or "")
+--					< string.lower(skins[b] and skins[b].name or "")
+--			end)
+		end
+
+		-- search through the saved vars and compare against the defaults
+		local found = nil
+		for index, value in pairs(TitanSkins) do
+			found = nil
+			-- See if the skin is a default one
+			for idx, v in pairs(TitanSkinsDefault) do
+				if TitanSkinsDefault[idx].name == TitanSkins[index].name then
+					found = idx
+				end
+			end
+			if found then
+				-- already inserted
+			else -- could be user placed or old Titan
+				if TitanSkins[index].titan then
+					-- old Titan skin - let it drop
+				else
+					-- assume it is a user installed skin
+					table.insert (skins, TitanSkins[index])
+--					table.sort(skins, function(a, b)
+--						return string.lower(skins[a] and skins[a].name or "")
+--							< string.lower(skins[b] and skins[b].name or "")
+--					end)
+				end
+			end
+		end
+		return skins
+	end
+end
+
+--[[ local
+NAME: Sync_panel_settings
+DESC: Routine to sync TitanPanelSettings - defaults to saved vars.
+VAR: Setting - table to use as the default
+OUT: None
+--]]
+local function Sync_panel_settings(settings)
+	-- Synchronize registered and saved variables
+--TitanDebug("Sync_1: "..(settings.FontName or "?"))
+	TitanVariables_SyncRegisterSavedVariables(settings, TitanPanelSettings)
+--TitanDebug("Sync_2: "..(TitanPanelSettings.FontName or "?"))
+end
+
+--[[ local
+NAME: Set_Timers
+DESC: Routine to reset / sync Titan settings.
+VAR:  None
+OUT:  None
+--]]
+local function Set_Timers(reset)
+	-- Titan is loaded so set the timers we want to use
+	TitanTimers = {
+		["EnterWorld"] = {obj = "PEW", callback = TitanPanel_AdjustFrames, delay = 4,},
+		["DualSpec"] = {obj = "SpecSwitch", callback = TitanPanel_AdjustFrames, delay = 2,},
+		["LDBRefresh"] = {obj = "LDB", callback = TitanLDBRefreshButton, delay = 2,},
+		["Adjust"] = {obj = "MoveAdj", callback = TitanPanel_AdjustFrames, delay = 1,},
+		["Vehicle"] = {obj = "Vehicle", callback = TitanPanel_AdjustFrames, delay = 1,},
+	}
+
+	if reset then
+		TitanAllSetVar("TimerPEW", TitanTimers["EnterWorld"].delay)
+		TitanAllSetVar("TimerDualSpec", TitanTimers["DualSpec"].delay)
+		TitanAllSetVar("TimerLDB", TitanTimers["LDBRefresh"].delay)
+		TitanAllSetVar("TimerAdjust", TitanTimers["Adjust"].delay)
+		TitanAllSetVar("TimerVehicle", TitanTimers["Vehicle"].delay)
+	else
+		TitanTimers["EnterWorld"].delay = TitanAllGetVar("TimerPEW")
+		TitanTimers["DualSpec"].delay = TitanAllGetVar("TimerDualSpec")
+		TitanTimers["LDBRefresh"].delay = TitanAllGetVar("TimerLDB")
+		TitanTimers["Adjust"].delay = TitanAllGetVar("TimerAdjust")
+		TitanTimers["Vehicle"].delay = TitanAllGetVar("TimerVehicle")
+	end
+end
+
+--[[ Titan
+NAME: TitanVariables_SyncPluginSettings
+DESC: Routine to sync plugin datas - current loaded (lua file) to any plugin saved vars (last save to disk).
+VAR:  None
+OUT:  None
+--]]
+function TitanVariables_SyncPluginSettings() -- one plugin uses this
+	-- Init each and every plugin
+	for id, plugin in pairs(TitanPlugins) do
+		if (plugin and plugin.savedVariables) then
+			-- Init savedVariables table
+			if (not TitanPluginSettings[id]) then
+				TitanPluginSettings[id] = {};
+			end
+
+			-- Synchronize registered and saved variables
+			TitanVariables_SyncRegisterSavedVariables(
+				plugin.savedVariables, TitanPluginSettings[id]);
+		else
+			-- Remove plugin savedVariables table if there's one
+			if (TitanPluginSettings[id]) then
+				TitanPluginSettings[id] = nil;
+			end
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanVariables_ExtraPluginSettings
+DESC: Routine to mark plugin data that is not loaded (no lua file) but has plugin saved vars (last save to disk).
+VAR:  None
+OUT:  None
+NOTE: This data is made available in case the user wants to delete the data via Tian Extras option.
+:NOTE
+--]]
+function TitanVariables_ExtraPluginSettings()
+	TitanPluginExtrasNum = 0
+	TitanPluginExtras = {}
+	-- Get the saved plugins that are not loaded
+	for id, plugin in pairs(TitanPluginSettings) do
+		if (id and TitanUtils_IsPluginRegistered(id)) then
+		else
+			TitanRegisterExtra(id)
+		end
+	end
+end
+
+--[[ Titan
+NAME: TitanVariables_InitTitanSettings
+DESC: Ensure TitanSettings (one of the saved vars in the toc) exists and set the Titan version.
+VAR:  None
+OUT:  None
+NOTE:
+- Called when Titan is loaded (ADDON_LOADED event)
+:NOTE
+--]]
+function TitanVariables_InitTitanSettings()
+	if (TitanSettings) then
+		-- all is good
+	else
+		TitanSettings = {}
+	end
+	-- check for player list per issue #745
+	if TitanSettings.Players then
+		-- all is good
+	else
+		-- Create the table so profile(s) can be added
+		TitanSettings.Players = {}
+	end
+--[[
+TitanDumpPlayerList()
+--]]
+	Sync_panel_settings(TITAN_PANEL_SAVED_VARIABLES)
+
+	if (TitanAll) then
+	else
+		TitanAll = {};
+	end
+	TitanVariables_SyncRegisterSavedVariables(TITAN_ALL_SAVED_VARIABLES, TitanAll)
+
+	TitanSettings.Version = TITAN_VERSION;
+end
+
+--[[ local
+NAME: Init_player_settings
+DESC: Use the Titan settings, the plugin settings, the 'extras' data of the given profile. Create the "to" profile if it does not exist.
+VAR:  from_profile - nil or profile to switch from (string)
+VAR:  to_profile - the toon to use (string)
+OUT:  None
+NOTE:
+- Called at PLAYER_ENTERING_WORLD event after we know Titan has registered plugins.
+- There are 3 actions: USE, RESET, and INIT
+- USE:
+ From: the user chosen profile
+ To: Player or Global profile
+- RESET:
+ From: Titan defaults
+ To: Player or Global profile
+- INIT:
+ From: saved variables of that profile
+ To: Player or Global profile
+:NOTE
+--]]
+local function Init_player_settings(from_profile, to_profile, action)
+	local old_player = {}
+	local old_panel = {}
+	local old_plugins = {}
+	local reset = (action == TITAN_PROFILE_RESET)
+--[[
+TitanDebug("_UseSettings "
+.."from: "..(from_profile or "?").." "
+.."to_profile: "..(to_profile or "?").." "
+.."action: "..action.." "
+)
+--]]
+
+	CleanupProfile () -- hide currently shown plugins
+	-- Ensure the requested profile is at least an empty stub
+	if (not TitanSettings.Players[to_profile]) or reset then
+		TitanSettings.Players[to_profile] = {}
+		TitanSettings.Players[to_profile].Plugins = {}
+		TitanSettings.Players[to_profile].Panel = {}
+		TitanSettings.Players[to_profile].Panel.Buttons = {}
+		TitanSettings.Players[to_profile].Panel.Location = {}
+		TitanPlayerSettings = {}
+		TitanPlayerSettings["Plugins"] = {}
+		TitanPlayerSettings["Panel"] = {}
+		TitanPlayerSettings["Register"] = {}
+	end
+	-- Set global variables
+	TitanPlayerSettings = TitanSettings.Players[to_profile];
+	TitanPluginSettings = TitanPlayerSettings["Plugins"];
+	TitanPanelSettings = TitanPlayerSettings["Panel"];
+	Sync_panel_settings(TITAN_PANEL_SAVED_VARIABLES);
+
+	if action == TITAN_PROFILE_RESET then
+		-- default is global profile OFF
+		TitanAll = {}
+		TitanVariables_SyncRegisterSavedVariables(TITAN_ALL_SAVED_VARIABLES, TitanAll)
+	elseif action == TITAN_PROFILE_INIT then
+		--
+	elseif action == TITAN_PROFILE_USE then
+		-- The requested profile at least exists so we can copy to it
+		-- Copy from the from_profile to profile - not anything in saved vars
+		if from_profile and TitanSettings.Players[from_profile] then
+			old_player = TitanSettings.Players[from_profile]
+		else
+		end
+		if old_player and old_player["Panel"] then
+			old_panel = old_player["Panel"]
+		end
+		if old_player and old_player["Plugins"] then
+			old_plugins = old_player["Plugins"]
+		end
+		-- Copy the panel settings
+		for index, id in pairs(old_panel) do
+			TitanPanelSetVar(index, old_panel[index]);
+		end
+		-- Copy the plugin settings
+		for plugin, i in pairs(old_plugins) do
+			for var, id in pairs(old_plugins[plugin]) do
+				TitanSetVar(plugin, var, old_plugins[plugin][var])
+			end
+		end
+	end
+
+	if (TitanPlayerSettings) then
+		-- Synchronize plugin settings with plugins that were registered
+		TitanVariables_SyncPluginSettings()
+		-- Display the plugins the user selected AND are registered
+		if reset then
+			Plugin_settings(reset)
+		else
+			TitanVariables_PluginSettingsInit()
+		end
+		TitanVariables_ExtraPluginSettings()
+	end
+
+	TitanSkins = TitanVariables_SyncSkins()
+
+	Set_Timers(reset)
+	-- for debug if a user needs to send in the Titan saved vars
+	TitanPlayerSettings["Register"] = {}
+	TitanPanelRegister = TitanPlayerSettings["Register"]
+
+	TitanSettings.Profile = to_profile
+end
+
+--[[ API
+NAME: TitanGetVar
+DESC: Get the value of the requested plugin variable.
+VAR: id - the plugin name (string)
+VAR: var - the name (string) of the variable
+OUT:  None
+NOTE:
+- 'var' is from the plugin <button>.registry.savedVariables table as created in the plugin lua.
+:NOTE
+--]]
+function TitanGetVar(id, var)
+	if (id and var and TitanPluginSettings and TitanPluginSettings[id]) then
+		-- compatibility check
+		if TitanPluginSettings[id][var] == "Titan Nil" then
+			TitanPluginSettings[id][var] = false
+		end
+		return TitanPluginSettings[id][var];
+		--return TitanUtils_Ternary(TitanPluginSettings[id][var] == false, nil, TitanPluginSettings[id][var]);
+	end
+end
+
+--[[ API
+NAME: TitanVarExists
+DESC: Determine if requested plugin variable exists.
+VAR: id - the plugin name (string)
+VAR: var - the name (string) of the variable
+OUT:  None
+NOTE:
+- 'var' is from the plugin <button>.registry.savedVariables table as created in the plugin lua.
+- This checks existence NOT false!
+:NOTE
+--]]
+function TitanVarExists(id, var)
+	-- We need to check for existance not true!
+	-- If the value is nil then it will not exist...
+	if (id and var and TitanPluginSettings and TitanPluginSettings[id]
+	and (TitanPluginSettings[id][var]
+		or TitanPluginSettings[id][var] == false) )
+	then
+		return true
+	else
+		return false
+	end
+end
+
+--[[ API
+NAME: TitanSetVar
+DESC: Get the value of the requested plugin variable to the given value.
+VAR: id - the plugin name (string)
+VAR: var - the name (string) of the variable
+VAR: value - new value of var
+OUT:  None
+NOTE:
+- 'var' is from the plugin <button>.registry.savedVariables table as created in the plugin lua.
+:NOTE
+--]]
+function TitanSetVar(id, var, value)
+	if (id and var and TitanPluginSettings and TitanPluginSettings[id]) then
+		TitanPluginSettings[id][var] = TitanUtils_Ternary(value, value, false);
+	end
+end
+
+--[[ API
+NAME: TitanToggleVar
+DESC: Toggle the value of the requested plugin variable. This assumes var value represents a boolean
+VAR: id - the plugin name (string)
+VAR: var - the name (string) of the variable
+OUT:  None
+NOTE:
+- Boolean in this case could be true / false or non zero / zero or nil.
+:NOTE
+--]]
+function TitanToggleVar(id, var)
+	if (id and var and TitanPluginSettings and TitanPluginSettings[id]) then
+		TitanSetVar(id, var, TitanUtils_Toggle(TitanGetVar(id, var)));
+	end
+end
+
+--[[ API
+NAME: TitanPanelGetVar
+DESC: Get the value of the requested Titan variable.
+VAR: var - the name (string) of the variable
+OUT: value of the requested Titan variable
+NOTE:
+- 'var' is from the TitanPanelSettings[var].
+:NOTE
+--]]
+function TitanPanelGetVar(var)
+	if (var and TitanPanelSettings) then
+		if TitanPanelSettings[var] == "Titan Nil" then
+			TitanPanelSettings[var] = false
+		end
+		return TitanUtils_Ternary(TitanPanelSettings[var] == false, nil, TitanPanelSettings[var]);
+	end
+end
+
+--[[ API
+NAME: TitanPanelSetVar
+DESC: Set the value of the requested Titan variable.
+VAR: var - the name (string) of the variable
+VAR: value - new value of var
+OUT:  None
+NOTE:
+- 'var' is from the TitanPanelSettings[var].
+:NOTE
+--]]
+function TitanPanelSetVar(var, value)
+	if (var and TitanPanelSettings) then
+		TitanPanelSettings[var] = TitanUtils_Ternary(value, value, false);
+	end
+end
+
+--[[ API
+NAME: TitanPanelToggleVar
+DESC: Toggle the value of the requested Titan variable. This assumes var value represents a boolean
+VAR:  var - the name (string) of the variable
+OUT:  None
+NOTE:
+- Boolean in this case could be true / false or non zero / zero or nil.
+:NOTE
+--]]
+function TitanPanelToggleVar(var)
+	if (var and TitanPanelSettings) then
+		TitanPanelSetVar(var, TitanUtils_Toggle(TitanPanelGetVar(var)));
+	end
+end
+
+--[[ API
+NAME: TitanAllGetVar
+DESC: Get the value of the requested Titan global variable.
+VAR: var - the name (string) of the variable
+OUT: None
+NOTE:
+- 'var' is from the TitanAll[var].
+:NOTE
+--]]
+function TitanAllGetVar(var)
+	if (var and TitanAll) then
+		if TitanAll[var] == "Titan Nil" then
+			TitanAll[var] = false
+		end
+		return TitanUtils_Ternary(TitanAll[var] == false, nil, TitanAll[var]);
+	end
+end
+
+--[[ API
+NAME: TitanAllSetVar
+DESC: Set the value of the requested Titan global variable.
+VAR: var - the name (string) of the variable
+VAR: value - new value of var
+OUT:  None
+NOTE:
+- 'var' is from the TitanPanelSettings[var].
+:NOTE
+--]]
+function TitanAllSetVar(var, value)
+	if (var and TitanAll) then
+		TitanAll[var] = TitanUtils_Ternary(value, value, false);
+	end
+end
+
+--[[ API
+NAME: TitanAllToggleVar
+DESC: Toggle the value of the requested Titan global variable. This assumes var value represents a boolean
+VAR: var - the name (string) of the variable
+OUT: None
+NOTE:
+- Boolean in this case could be true / false or non zero / zero or nil.
+:NOTE
+--]]
+function TitanAllToggleVar(var)
+	if (var and TitanAll) then
+		TitanAllSetVar(var, TitanUtils_Toggle(TitanAllGetVar(var)));
+	end
+end
+
+--[[ API
+NAME: TitanVariables_GetPanelStrata
+DESC: Return the strata and the next highest strata of the given value
+VAR: value - the name (string) of the strata to look up
+OUT: string - Next highest strata
+OUT: string - passed in strata
+--]]
+function TitanVariables_GetPanelStrata(value)
+	-- obligatory check
+	if not value then value = "DIALOG" end
+
+	local index;
+	local indexpos = 5 -- DIALOG
+	local StrataTypes = {"BACKGROUND", "LOW", "MEDIUM", "HIGH",
+		"DIALOG", "FULLSCREEN", "FULLSCREEN_DIALOG"}
+
+	for index in ipairs(StrataTypes) do
+		if value == StrataTypes[index] then
+			indexpos = index
+			break
+		end
+	end
+
+	return StrataTypes[indexpos + 1], StrataTypes[indexpos]
+end
+
+--[[ API
+NAME: TitanVariables_SetPanelStrata
+DESC: Set the Titan bars to the given strata and the plugins to the next highest strata.
+VAR: value - strata name (string)
+OUT: None
+--]]
+function TitanVariables_SetPanelStrata(value)
+	local plugins, bars = TitanVariables_GetPanelStrata(value)
+	local idx, v
+	-- Set all the Titan bars
+	for idx,v in pairs (TitanBarData) do
+		local bar_name = TITAN_PANEL_DISPLAY_PREFIX..TitanBarData[idx].name
+		_G[bar_name]:SetFrameStrata(bars)
+	end
+	-- Set all the registered plugins
+	for idx, v in pairs(TitanPluginsIndex) do
+		local button = TitanUtils_GetButton(v);
+		button:SetFrameStrata(plugins)
+	end
+end
+
+--[[ Titan
+NAME: TitanVariables_UseSettings
+DESC: Set the Titan variables and plugin variables to the passed in profile.
+VAR: profile - profile to use for this toon : <name>@<server>
+OUT: None
+NOTE:
+- Called from the Titan right click menu
+- profile is compared as 'lower' so the case of profile does not matter
+:NOTE
+--]]
+function TitanVariables_UseSettings(profile, action)
+--[[
+TitanDebug("_UseSettings "
+.."profile: "..(profile or "?").." "
+.."action: "..action.." "
+)
+--]]
+	-- sanity checks to ensure the base tables are set
+	if (TitanSettings) then
+		-- all is good
+	else
+		TitanSettings = {}
+	end
+	-- check for player list per issue #745
+	if TitanSettings.Players then
+		-- all is good
+	else
+		-- Create the table so profile(s) can be added
+		TitanSettings.Players = {}
+	end
+	if (TitanAll) then
+		-- all is good
+	else
+		TitanAll = {};
+	end
+	TitanVariables_SyncRegisterSavedVariables(TITAN_ALL_SAVED_VARIABLES, TitanAll)
+
+	TitanSettings.Version = TITAN_VERSION;
+
+	local from_profile = nil
+	if action == TITAN_PROFILE_USE then
+		-- Grab the old profile currently in use
+		from_profile = profile or nil
+	end
+
+	local _ = nil
+	local glob, name, player, server = TitanUtils_GetGlobalProfile()
+	-- Get the profile according to the user settings
+	if glob then
+		profile = name -- Use global toon
+	else
+		profile, _, _ = TitanUtils_GetPlayer() -- Use current toon
+	end
+
+	-- Find the profile in a case insensitive manner
+	local new_profile = ""
+	profile = string.lower(profile)
+	for index, id in pairs(TitanSettings.Players) do
+		if profile == string.lower(index) then
+			new_profile = index
+		end
+	end
+	if new_profile == "" then
+		-- Assume we need the current player
+		new_profile = TitanUtils_GetPlayer() --TitanSettings.Player
+		-- And it needs to be created
+		action = TITAN_PROFILE_RESET
+	end
+
+	-- Now that we know what profile to use - act on the data
+	Init_player_settings(from_profile, new_profile, action)
+
+	-- set strata in case it has changed
+	TitanVariables_SetPanelStrata(TitanPanelGetVar("FrameStrata"))
+
+	-- show the new profile
+	TitanPanel_InitPanelBarButton();
+	TitanPanel_InitPanelButtons();
+end
+
+-- decrecated routines
+--[[
+
+function TitanGetVarTable(id, var, position)
+	if (id and var and TitanPluginSettings and TitanPluginSettings[id]) then
+		-- compatibility check
+		if TitanPluginSettings[id][var][position] == "Titan Nil" then TitanPluginSettings[id][var][position] = false end
+		return TitanUtils_Ternary(TitanPluginSettings[id][var][position] == false, nil, TitanPluginSettings[id][var][position]);
+	end
+end
+
+function TitanSetVarTable(id, var, position, value)
+	if (id and var and TitanPluginSettings and TitanPluginSettings[id]) then
+		TitanPluginSettings[id][var][position] = TitanUtils_Ternary(value, value, false);
+	end
+end
+
+--]]
\ No newline at end of file
diff --git a/Titan/embeds.xml b/Titan/embeds.xml
new file mode 100644
index 0000000..770d0b1
--- /dev/null
+++ b/Titan/embeds.xml
@@ -0,0 +1,16 @@
+<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\AceHook-3.0\AceHook-3.0.xml"/>
+<Include file="libs\AceTimer-3.0\AceTimer-3.0.xml"/>
+<Include file="libs\AceGUI-3.0\AceGUI-3.0.xml"/>
+<Include file="libs\AceConfig-3.0\AceConfig-3.0.xml"/>
+<Include file="libs\AceLocale-3.0\AceLocale-3.0.xml"/>
+<Include file="libs\LibSharedMedia-3.0\lib.xml"/>
+<Include file="libs\AceGUI-3.0-SharedMediaWidgets\widget.xml"/>
+<Include file="libs\LibQTip-1.0\lib.xml"/>
+<Include file="libs\!LibUIDropDownMenu\LibUIDropDownMenu\LibUIDropDownMenu.xml"/>
+</Ui>
diff --git a/Titan/libs/!LibUIDropDownMenu/!LibUIDropDownMenu.toc b/Titan/libs/!LibUIDropDownMenu/!LibUIDropDownMenu.toc
new file mode 100755
index 0000000..1570f6e
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/!LibUIDropDownMenu.toc
@@ -0,0 +1,12 @@
+## Interface: 80000
+## Title: Lib: UIDropDownMenu
+## Notes: A Replacement for standard UIDropDownMenu
+## Version: v1.08.80001273226
+## X-Category: Libraries
+## X-Website: https://www.wowace.com/projects/libuidropdownmenu
+## OptionalDeps: LibStub
+
+#@no-lib-strip@
+LibStub\LibStub.lua
+#@end-no-lib-strip@
+LibUIDropDownMenu\LibUIDropDownMenu.xml
diff --git a/Titan/libs/!LibUIDropDownMenu/Docs/Readme.txt b/Titan/libs/!LibUIDropDownMenu/Docs/Readme.txt
new file mode 100755
index 0000000..7e23b95
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/Docs/Readme.txt
@@ -0,0 +1,126 @@
+== About ==
+Standard UIDropDownMenu global functions using protected frames and causing taints
+when used by third-party addons. But it is possible to avoid taints by using same
+functionality with that library.
+
+== What is it ==
+Library is standard code from Blizzard's files EasyMenu.lua, UIDropDownMenu.lua,
+UIDropDownMenu.xml and UIDropDownMenuTemplates.xml with frames, tables, variables
+and functions renamed to:
+* constants : "L_" added at the start
+* functions: "L_" added at the start
+
+== How to use it (for addon developer) ==
+* Embed LibUIDropDownMenu to your addon, you can specify to the folder to
+  LibUIDropDownMenu\LibUIDropDownMenu if you feel this keep the folder cleaner.
+* Add LibUIDropDownMenu.xml to your toc or your embeds.xml / libs.xml.
+* If your addon doesn't embed LibStub, you will need it.
+* Like ordinal code for UIDropDownMenu with "L_" instead.
+
+== Constants ==
+* L_UIDROPDOWNMENU_MINBUTTONS
+* L_UIDROPDOWNMENU_MAXBUTTONS
+* L_UIDROPDOWNMENU_MAXLEVELS
+* L_UIDROPDOWNMENU_BUTTON_HEIGHT
+* L_UIDROPDOWNMENU_BORDER_HEIGHT
+* L_UIDROPDOWNMENU_OPEN_MENU
+* L_UIDROPDOWNMENU_INIT_MENU
+* L_UIDROPDOWNMENU_MENU_LEVEL
+* L_UIDROPDOWNMENU_MENU_VALUE
+* L_UIDROPDOWNMENU_SHOW_TIME
+* L_UIDROPDOWNMENU_DEFAULT_TEXT_HEIGHT
+* L_OPEN_DROPDOWNMENUS
+
+== Functions ==
+* L_EasyMenu
+* L_EasyMenu_Initialize
+
+* L_UIDropDownMenuDelegate_OnAttributeChanged
+* L_UIDropDownMenu_InitializeHelper
+* L_UIDropDownMenu_Initialize
+* L_UIDropDownMenu_SetInitializeFunction
+* L_UIDropDownMenu_RefreshDropDownSize
+* L_UIDropDownMenu_OnUpdate
+* L_UIDropDownMenu_StartCounting
+* L_UIDropDownMenu_StopCounting
+* L_UIDropDownMenu_CheckAddCustomFrame
+* L_UIDropDownMenu_CreateInfo
+* L_UIDropDownMenu_CreateFrames
+* L_UIDropDownMenu_AddSeparator
+* L_UIDropDownMenu_AddButton
+* L_UIDropDownMenu_AddSeparator
+* L_UIDropDownMenu_GetMaxButtonWidth
+* L_UIDropDownMenu_GetButtonWidth
+* L_UIDropDownMenu_Refresh
+* L_UIDropDownMenu_RefreshAll
+* L_UIDropDownMenu_RegisterCustomFrame
+* L_UIDropDownMenu_SetIconImage
+* L_UIDropDownMenu_SetSelectedName
+* L_UIDropDownMenu_SetSelectedValue
+* L_UIDropDownMenu_SetSelectedID
+* L_UIDropDownMenu_GetSelectedName
+* L_UIDropDownMenu_GetSelectedID
+* L_UIDropDownMenu_GetSelectedValue
+* L_UIDropDownMenuButton_OnClick
+* L_HideDropDownMenu
+* L_ToggleDropDownMenu
+* L_CloseDropDownMenus
+* L_UIDropDownMenu_OnHide
+* L_UIDropDownMenu_SetWidth
+* L_UIDropDownMenu_SetButtonWidth
+* L_UIDropDownMenu_SetText
+* L_UIDropDownMenu_GetText
+* L_UIDropDownMenu_ClearAll
+* L_UIDropDownMenu_JustifyText
+* L_UIDropDownMenu_SetAnchor
+* L_UIDropDownMenu_GetCurrentDropDown
+* L_UIDropDownMenuButton_GetChecked
+* L_UIDropDownMenuButton_GetName
+* L_UIDropDownMenuButton_OpenColorPicker
+* L_UIDropDownMenu_DisableButton
+* L_UIDropDownMenu_EnableButton
+* L_UIDropDownMenu_SetButtonText
+* L_UIDropDownMenu_SetButtonNotClickable
+* L_UIDropDownMenu_SetButtonClickable
+* L_UIDropDownMenu_DisableDropDown
+* L_UIDropDownMenu_EnableDropDown
+* L_UIDropDownMenu_IsEnabled
+* L_UIDropDownMenu_GetValue
+
+== List of button attributes ==
+* info.text = [STRING]  --  The text of the button
+* info.value = [ANYTHING]  --  The value that L_UIDROPDOWNMENU_MENU_VALUE is set to when the button is clicked
+* info.func = [function()]  --  The function that is called when you click the button
+* info.checked = [nil, true, function]  --  Check the button if true or function returns true
+* info.isNotRadio = [nil, true]  --  Check the button uses radial image if false check box image if true
+* info.isTitle = [nil, true]  --  If it's a title the button is disabled and the font color is set to yellow
+* info.disabled = [nil, true]  --  Disable the button and show an invisible button that still traps the mouseover event so menu doesn't time out
+* info.tooltipWhileDisabled = [nil, 1] -- Show the tooltip, even when the button is disabled.
+* info.hasArrow = [nil, true]  --  Show the expand arrow for multilevel menus
+* info.hasColorSwatch = [nil, true]  --  Show color swatch or not, for color selection
+* info.r = [1 - 255]  --  Red color value of the color swatch
+* info.g = [1 - 255]  --  Green color value of the color swatch
+* info.b = [1 - 255]  --  Blue color value of the color swatch
+* info.colorCode = [STRING] -- "|cAARRGGBB" embedded hex value of the button text color. Only used when button is enabled
+* info.swatchFunc = [function()]  --  Function called by the color picker on color change
+* info.hasOpacity = [nil, 1]  --  Show the opacity slider on the colorpicker frame
+* info.opacity = [0.0 - 1.0]  --  Percentatge of the opacity, 1.0 is fully shown, 0 is transparent
+* info.opacityFunc = [function()]  --  Function called by the opacity slider when you change its value
+* info.cancelFunc = [function(previousValues)] -- Function called by the colorpicker when you click the cancel button (it takes the previous values as its argument)
+* info.notClickable = [nil, 1]  --  Disable the button and color the font white
+* info.notCheckable = [nil, 1]  --  Shrink the size of the buttons and don't display a check box
+* info.owner = [Frame]  --  Dropdown frame that "owns" the current dropdownlist
+* info.keepShownOnClick = [nil, 1]  --  Don't hide the dropdownlist after a button is clicked
+* info.tooltipTitle = [nil, STRING] -- Title of the tooltip shown on mouseover
+* info.tooltipText = [nil, STRING] -- Text of the tooltip shown on mouseover
+* info.tooltipOnButton = [nil, 1] -- Show the tooltip attached to the button instead of as a Newbie tooltip.
+* info.justifyH = [nil, "CENTER"] -- Justify button text
+* info.arg1 = [ANYTHING] -- This is the first argument used by info.func
+* info.arg2 = [ANYTHING] -- This is the second argument used by info.func
+* info.fontObject = [FONT] -- font object replacement for Normal and Highlight
+* info.menuTable = [TABLE] -- This contains an array of info tables to be displayed as a child menu
+* info.noClickSound = [nil, 1]  --  Set to 1 to suppress the sound when clicking the button. The sound only plays if .func is set.
+* info.padding = [nil, NUMBER] -- Number of pixels to pad the text on the right side
+* info.leftPadding = [nil, NUMBER] -- Number of pixels to pad the button on the left side
+* info.minWidth = [nil, NUMBER] -- Minimum width for this line
+* info.customFrame = frame -- Allows this button to be a completely custom frame, should inherit from L_UIDropDownCustomMenuEntryTemplate and override appropriate methods.
\ No newline at end of file
diff --git a/Titan/libs/!LibUIDropDownMenu/Docs/Revision.txt b/Titan/libs/!LibUIDropDownMenu/Docs/Revision.txt
new file mode 100755
index 0000000..1b99a8d
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/Docs/Revision.txt
@@ -0,0 +1,76 @@
+$Id: Revision.txt 36 2018-08-11 13:29:16Z arith $
+Revision History:
+=================
+
+v1.08.8000127326 (2018/08/11)
+-----------------------------
+- Workaround to get rid of addons which are still using old version of this library
+  (thanks to ddcorkum)
+
+v1.08.8000127165 (2018/07/25)
+-----------------------------
+- Sync with WoW 8.0.1.27165
+  - Added L_UIDropDownMenu_SetDisplayMode()
+  - Added L_UIDropDownMenuButtonInvisibleButton_OnEnter()
+  - Added L_UIDropDownMenuButtonInvisibleButton_OnLeave()
+  - Added L_UIDropDownMenuButton_OnEnter()
+  - Added L_UIDropDownMenuButton_OnLeave()
+
+v1.08.8000126433 (2018/04/24)
+-----------------------------
+- Sync with WoW 8.0.1.26433
+  - Added LibUIDropDownMenuTemplates.lua
+  - Added frame template: L_UIDropDownCustomMenuEntryTemplate
+  - Added local function GetChild()
+  - New custom frame functions:
+    - L_UIDropDownMenu_CheckAddCustomFrame()
+	- L_UIDropDownMenu_RegisterCustomFrame()
+  - New button attribute: info.customFrame
+  - Changes of L_UIDropDownMenu_AddSeparator() is to be reflected only on WoW 8.x.x
+    - L_UIDropDownMenu_AddSeparator(info, level) == > L_UIDropDownMenu_AddSeparator(level)
+	  No need to specify info there to prevent from messing up other menu items if info is to be re-used.
+- Fixed the lib's major version.
+  Previously with the wrong major version which also has the release version, it makes all the different versions of lib to be presented
+  as different entities, which means newer version won't replace the older version. The latest loaded one will replace all the constants
+  and functions.
+  This change will take effect until all the addons which embed this lib to replace with latest version.
+
+v1.07.7030525961 (2018/04/23)
+-----------------------------
+- Remove external
+- Refine upvalue and local function pre-definition
+- Rename local parameters to make them consistent with others
+
+v1.07.7030024931 (2017/08/31)
+-----------------------------
+- Remove PlaySound compact code.
+
+v1.06.7030024931 (2017/08/31)
+-----------------------------
+- Remove PlaySound compact code.
+
+v1.05.7030024920 (2017/08/29)
+-----------------------------
+- ToC update to support WoW 7.3.0
+
+v1.04.7030024484 (2017/07/02)
+-----------------------------
+- Changed soundkit's ID to key indicator likes SOUNDKIT.U_CHAT_SCROLL_BUTTON
+
+v1.03.7030024484 (2017/06/30)
+-----------------------------
+- Update version number
+
+v1.02.7030024484 (2017/06/30)
+-----------------------------
+- Updated PlaySound API's usage method to support both 7.2.5 and 7.3.0 of WoW clients
+
+v1.01.7020024015 (2017/05/25)
+----------------------------
+- Fixed the wrong name of LibEasyMenu.lua specified in LibUIDropDownMenu.xml
+
+v1.00.7020024015 (2017/05/24)
+----------------------------
+- Clone from NoTaint_UIDropDownMenu v7.2.0.24015-r2
+- Changed LIB_ and Lib_ to L_ to prevent from conflict with outdated
+  NoTaint_UIDropDownMenu being loaded from other addons
diff --git a/Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.lua b/Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.lua
new file mode 100755
index 0000000..7e7b76d
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.lua
@@ -0,0 +1,51 @@
+-- $Id: LibStub.lua 103 2014-10-16 03:02:50Z mikk $
+-- LibStub is a simple versioning stub meant for use in Libraries.  http://www.wowace.com/addons/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]
+
+-- Check to see is this version of the stub is obsolete
+if not LibStub or LibStub.minor < LIBSTUB_MINOR then
+	LibStub = LibStub or {libs = {}, minors = {} }
+	_G[LIBSTUB_MAJOR] = LibStub
+	LibStub.minor = LIBSTUB_MINOR
+
+	-- LibStub:NewLibrary(major, minor)
+	-- major (string) - the major version of the library
+	-- minor (string or number ) - the minor version of the library
+	--
+	-- returns nil if a newer or same version of the lib is already present
+	-- returns empty library object or old library object if upgrade is needed
+	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
+
+	-- LibStub:GetLibrary(major, [silent])
+	-- major (string) - the major version of the library
+	-- silent (boolean) - if true, library is optional, silently return nil if its not found
+	--
+	-- throws an error if the library can not be found (except silent is set)
+	-- returns the library object if found
+	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
+
+	-- LibStub:IterateLibraries()
+	--
+	-- Returns an iterator for the currently registered libraries
+	function LibStub:IterateLibraries()
+		return pairs(self.libs)
+	end
+
+	setmetatable(LibStub, { __call = LibStub.GetLibrary })
+end
diff --git a/Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.toc b/Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.toc
new file mode 100755
index 0000000..31aef48
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibStub/LibStub.toc
@@ -0,0 +1,9 @@
+## Interface: 70200
+## Title: Lib: LibStub
+## Notes: Universal Library Stub
+## Credits: Kaelten, Cladhaire, ckknight, Mikk, Ammo, Nevcairiel
+## X-Website: http://www.wowace.com/addons/libstub/
+## X-Category: Library
+## X-License: Public Domain
+
+LibStub.lua
diff --git a/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test.lua b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test.lua
new file mode 100755
index 0000000..645a08b
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test.lua
@@ -0,0 +1,41 @@
+debugstack = debug.traceback
+strmatch = string.match
+
+loadfile("../LibStub.lua")()
+
+local lib, oldMinor = LibStub:NewLibrary("Pants", 1) -- make a new thingy
+assert(lib) -- should return the library table
+assert(not oldMinor) -- should not return the old minor, since it didn't exist
+
+-- the following is to create data and then be able to check if the same data exists after the fact
+function lib:MyMethod()
+end
+local MyMethod = lib.MyMethod
+lib.MyTable = {}
+local MyTable = lib.MyTable
+
+local newLib, newOldMinor = LibStub:NewLibrary("Pants", 1) -- try to register a library with the same version, should silently fail
+assert(not newLib) -- should not return since out of date
+
+local newLib, newOldMinor = LibStub:NewLibrary("Pants", 0) -- try to register a library with a previous, should silently fail
+assert(not newLib) -- should not return since out of date
+
+local newLib, newOldMinor = LibStub:NewLibrary("Pants", 2) -- register a new version
+assert(newLib) -- library table
+assert(rawequal(newLib, lib)) -- should be the same reference as the previous
+assert(newOldMinor == 1) -- should return the minor version of the previous version
+
+assert(rawequal(lib.MyMethod, MyMethod)) -- verify that values were saved
+assert(rawequal(lib.MyTable, MyTable)) -- verify that values were saved
+
+local newLib, newOldMinor = LibStub:NewLibrary("Pants", "Blah 3 Blah") -- register a new version with a string minor version (instead of a number)
+assert(newLib) -- library table
+assert(newOldMinor == 2) -- previous version was 2
+
+local newLib, newOldMinor = LibStub:NewLibrary("Pants", "Blah 4 and please ignore 15 Blah") -- register a new version with a string minor version (instead of a number)
+assert(newLib)
+assert(newOldMinor == 3) -- previous version was 3 (even though it gave a string)
+
+local newLib, newOldMinor = LibStub:NewLibrary("Pants", 5) -- register a new library, using a normal number instead of a string
+assert(newLib)
+assert(newOldMinor == 4) -- previous version was 4 (even though it gave a string)
\ No newline at end of file
diff --git a/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test2.lua b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test2.lua
new file mode 100755
index 0000000..af431dd
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test2.lua
@@ -0,0 +1,27 @@
+debugstack = debug.traceback
+strmatch = string.match
+
+loadfile("../LibStub.lua")()
+
+for major, library in LibStub:IterateLibraries() do
+	-- check that MyLib doesn't exist yet, by iterating through all the libraries
+	assert(major ~= "MyLib")
+end
+
+assert(not LibStub:GetLibrary("MyLib", true)) -- check that MyLib doesn't exist yet by direct checking
+assert(not pcall(LibStub.GetLibrary, LibStub, "MyLib")) -- don't silently fail, thus it should raise an error.
+local lib = LibStub:NewLibrary("MyLib", 1) -- create the lib
+assert(lib) -- check it exists
+assert(rawequal(LibStub:GetLibrary("MyLib"), lib)) -- verify that :GetLibrary("MyLib") properly equals the lib reference
+
+assert(LibStub:NewLibrary("MyLib", 2))	-- create a new version
+
+local count=0
+for major, library in LibStub:IterateLibraries() do
+	-- check that MyLib exists somewhere in the libraries, by iterating through all the libraries
+	if major == "MyLib" then -- we found it!
+		count = count +1
+		assert(rawequal(library, lib)) -- verify that the references are equal
+	end
+end
+assert(count == 1) -- verify that we actually found it, and only once
diff --git a/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test3.lua b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test3.lua
new file mode 100755
index 0000000..01aabb8
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test3.lua
@@ -0,0 +1,14 @@
+debugstack = debug.traceback
+strmatch = string.match
+
+loadfile("../LibStub.lua")()
+
+local proxy = newproxy() -- non-string
+
+assert(not pcall(LibStub.NewLibrary, LibStub, proxy, 1)) -- should error, proxy is not a string, it's userdata
+local success, ret = pcall(LibStub.GetLibrary, proxy, true)
+assert(not success or not ret) -- either error because proxy is not a string or because it's not actually registered.
+
+assert(not pcall(LibStub.NewLibrary, LibStub, "Something", "No number in here")) -- should error, minor has no string in it.
+
+assert(not LibStub:GetLibrary("Something", true)) -- shouldn't've created it from the above statement
\ No newline at end of file
diff --git a/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test4.lua b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test4.lua
new file mode 100755
index 0000000..15a9c9c
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibStub/tests/test4.lua
@@ -0,0 +1,41 @@
+debugstack = debug.traceback
+strmatch = string.match
+
+loadfile("../LibStub.lua")()
+
+
+-- Pretend like loaded libstub is old and doesn't have :IterateLibraries
+assert(LibStub.minor)
+LibStub.minor = LibStub.minor - 0.0001
+LibStub.IterateLibraries = nil
+
+loadfile("../LibStub.lua")()
+
+assert(type(LibStub.IterateLibraries)=="function")
+
+
+-- Now pretend that we're the same version -- :IterateLibraries should NOT be re-created
+LibStub.IterateLibraries = 123
+
+loadfile("../LibStub.lua")()
+
+assert(LibStub.IterateLibraries == 123)
+
+
+-- Now pretend that a newer version is loaded -- :IterateLibraries should NOT be re-created
+LibStub.minor = LibStub.minor + 0.0001
+
+loadfile("../LibStub.lua")()
+
+assert(LibStub.IterateLibraries == 123)
+
+
+-- Again with a huge number
+LibStub.minor = LibStub.minor + 1234567890
+
+loadfile("../LibStub.lua")()
+
+assert(LibStub.IterateLibraries == 123)
+
+
+print("OK")
\ No newline at end of file
diff --git a/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibEasyMenu.lua b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibEasyMenu.lua
new file mode 100755
index 0000000..5cf69f6
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibEasyMenu.lua
@@ -0,0 +1,47 @@
+--$Id: LibEasyMenu.lua 30 2018-04-24 06:44:39Z arith $
+-- Simplified Menu Display System
+--	This is a basic system for displaying a menu from a structure table.
+--
+--	See UIDropDownMenu.lua for the menuList details.
+--
+--	Args:
+--		menuList - menu table
+--		menuFrame - the UI frame to populate
+--		anchor - where to anchor the frame (e.g. CURSOR)
+--		x - x offset
+--		y - y offset
+--		displayMode - border type
+--		autoHideDelay - how long until the menu disappears
+--
+--
+-- ----------------------------------------------------------------------------
+-- Localized Lua globals.
+-- ----------------------------------------------------------------------------
+local _G = getfenv(0)
+-- ----------------------------------------------------------------------------
+local MAJOR_VERSION = "LibEasyMenu"
+local MINOR_VERSION = 90000 + tonumber(("$Rev: 30 $"):match("%d+"))
+
+local LibStub = _G.LibStub
+if not LibStub then error(MAJOR_VERSION .. " requires LibStub.") end
+local Lib = LibStub:NewLibrary(MAJOR_VERSION, MINOR_VERSION)
+if not Lib then return end
+
+function L_EasyMenu(menuList, menuFrame, anchor, x, y, displayMode, autoHideDelay )
+	if ( displayMode == "MENU" ) then
+		menuFrame.displayMode = displayMode;
+	end
+	L_UIDropDownMenu_Initialize(menuFrame, L_EasyMenu_Initialize, displayMode, nil, menuList);
+	L_ToggleDropDownMenu(1, nil, menuFrame, anchor, x, y, menuList, nil, autoHideDelay);
+end
+
+function L_EasyMenu_Initialize( frame, level, menuList )
+	for index = 1, #menuList do
+		local value = menuList[index]
+		if (value.text) then
+			value.index = index;
+			L_UIDropDownMenu_AddButton( value, level );
+		end
+	end
+end
+
diff --git a/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.lua b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.lua
new file mode 100755
index 0000000..de3b658
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.lua
@@ -0,0 +1,1411 @@
+-- $Id: LibUIDropDownMenu.lua 36 2018-08-11 13:29:16Z arith $
+-- ----------------------------------------------------------------------------
+-- Localized Lua globals.
+-- ----------------------------------------------------------------------------
+local _G = getfenv(0)
+local tonumber, type, string, table = tonumber, type, string, table
+local strsub, strlen, strmatch, gsub = strsub, strlen, strmatch, gsub
+local max, match = max, match
+local securecall, issecure = securecall, issecure
+local wipe = table.wipe
+-- WoW
+local CreateFrame, GetCursorPosition, GetCVar, GetScreenHeight, GetScreenWidth, OpenColorPicker, PlaySound = CreateFrame, GetCursorPosition, GetCVar, GetScreenHeight, GetScreenWidth, OpenColorPicker, PlaySound
+
+-- ----------------------------------------------------------------------------
+local WRONG_VERSION = "LibUIDropDownMenu-1.07.7030024931"
+local MAJOR_VERSION = "LibUIDropDownMenu"
+local MINOR_VERSION = 90000 + tonumber(("$Rev: 36 $"):match("%d+"))
+
+local LibStub = _G.LibStub
+if not LibStub then error(MAJOR_VERSION .. " requires LibStub.") end
+local wronglib = LibStub:NewLibrary(WRONG_VERSION, MINOR_VERSION)
+local lib = LibStub:NewLibrary(MAJOR_VERSION, MINOR_VERSION)
+if not lib then return end
+
+-- //////////////////////////////////////////////////////////////
+L_UIDROPDOWNMENU_MINBUTTONS = 8;
+L_UIDROPDOWNMENU_MAXBUTTONS = 8;
+L_UIDROPDOWNMENU_MAXLEVELS = 2;
+L_UIDROPDOWNMENU_BUTTON_HEIGHT = 16;
+L_UIDROPDOWNMENU_BORDER_HEIGHT = 15;
+-- The current open menu
+L_UIDROPDOWNMENU_OPEN_MENU = nil;
+-- The current menu being initialized
+L_UIDROPDOWNMENU_INIT_MENU = nil;
+-- Current level shown of the open menu
+L_UIDROPDOWNMENU_MENU_LEVEL = 1;
+-- Current value of the open menu
+L_UIDROPDOWNMENU_MENU_VALUE = nil;
+-- Time to wait to hide the menu
+L_UIDROPDOWNMENU_SHOW_TIME = 2;
+-- Default dropdown text height
+L_UIDROPDOWNMENU_DEFAULT_TEXT_HEIGHT = nil;
+-- List of open menus
+L_OPEN_DROPDOWNMENUS = {};
+
+local L_UIDropDownMenuDelegate = CreateFrame("FRAME");
+
+function L_UIDropDownMenuDelegate_OnAttributeChanged (self, attribute, value)
+	if ( attribute == "createframes" and value == true ) then
+		L_UIDropDownMenu_CreateFrames(self:GetAttribute("createframes-level"), self:GetAttribute("createframes-index"));
+	elseif ( attribute == "initmenu" ) then
+		L_UIDROPDOWNMENU_INIT_MENU = value;
+	elseif ( attribute == "openmenu" ) then
+		L_UIDROPDOWNMENU_OPEN_MENU = value;
+	end
+end
+
+L_UIDropDownMenuDelegate:SetScript("OnAttributeChanged", L_UIDropDownMenuDelegate_OnAttributeChanged);
+
+function L_UIDropDownMenu_InitializeHelper (frame)
+	-- This deals with the potentially tainted stuff!
+	if ( frame ~= L_UIDROPDOWNMENU_OPEN_MENU ) then
+		L_UIDROPDOWNMENU_MENU_LEVEL = 1;
+	end
+
+	-- Set the frame that's being intialized
+	L_UIDropDownMenuDelegate:SetAttribute("initmenu", frame);
+
+	-- Hide all the buttons
+	local button, dropDownList;
+	for i = 1, L_UIDROPDOWNMENU_MAXLEVELS, 1 do
+		dropDownList = _G["L_DropDownList"..i];
+		if ( i >= L_UIDROPDOWNMENU_MENU_LEVEL or frame ~= L_UIDROPDOWNMENU_OPEN_MENU ) then
+			dropDownList.numButtons = 0;
+			dropDownList.maxWidth = 0;
+			for j=1, L_UIDROPDOWNMENU_MAXBUTTONS, 1 do
+				button = _G["L_DropDownList"..i.."Button"..j];
+				button:Hide();
+			end
+			dropDownList:Hide();
+		end
+	end
+	frame:SetHeight(L_UIDROPDOWNMENU_BUTTON_HEIGHT * 2);
+end
+
+-- added since 8.0.1.26433
+local function GetChild(frame, name, key)
+	if (frame[key]) then
+		return frame[key];
+	else
+		return _G[name..key];
+	end
+end
+
+function L_UIDropDownMenu_Initialize(frame, initFunction, displayMode, level, menuList)
+	frame.menuList = menuList;
+
+	securecall("L_UIDropDownMenu_InitializeHelper", frame);
+
+	-- Set the initialize function and call it.  The initFunction populates the dropdown list.
+	if ( initFunction ) then
+		L_UIDropDownMenu_SetInitializeFunction(frame, initFunction);
+		initFunction(frame, level, frame.menuList);
+	end
+
+	--master frame
+	if(level == nil) then
+		level = 1;
+	end
+
+	local dropDownList = _G["L_DropDownList"..level];
+	dropDownList.dropdown = frame;
+	dropDownList.shouldRefresh = true;
+
+	L_UIDropDownMenu_SetDisplayMode(frame, displayMode);
+end
+
+function L_UIDropDownMenu_SetInitializeFunction(frame, initFunction)
+	frame.initialize = initFunction;
+end
+
+function L_UIDropDownMenu_SetDisplayMode(frame, displayMode)
+	-- Change appearance based on the displayMode
+	-- Note: this is a one time change based on previous behavior.
+	if ( displayMode == "MENU" ) then
+		local name = frame:GetName();
+		GetChild(frame, name, "Left"):Hide();
+		GetChild(frame, name, "Middle"):Hide();
+		GetChild(frame, name, "Right"):Hide();
+		local button = GetChild(frame, name, "Button");
+		local buttonName = button:GetName();
+		GetChild(button, buttonName, "NormalTexture"):SetTexture(nil);
+		GetChild(button, buttonName, "DisabledTexture"):SetTexture(nil);
+		GetChild(button, buttonName, "PushedTexture"):SetTexture(nil);
+		GetChild(button, buttonName, "HighlightTexture"):SetTexture(nil);
+		local text = GetChild(frame, name, "Text");
+
+		button:ClearAllPoints();
+		button:SetPoint("LEFT", text, "LEFT", -9, 0);
+		button:SetPoint("RIGHT", text, "RIGHT", 6, 0);
+		frame.displayMode = "MENU";
+	end
+end
+
+function L_UIDropDownMenu_RefreshDropDownSize(self)
+	self.maxWidth = L_UIDropDownMenu_GetMaxButtonWidth(self);
+	self:SetWidth(self.maxWidth + 25);
+
+	for i=1, L_UIDROPDOWNMENU_MAXBUTTONS, 1 do
+		local icon = _G[self:GetName().."Button"..i.."Icon"];
+
+		if ( icon.tFitDropDownSizeX ) then
+			icon:SetWidth(self.maxWidth - 5);
+		end
+	end
+end
+
+-- If dropdown is visible then see if its timer has expired, if so hide the frame
+function L_UIDropDownMenu_OnUpdate(self, elapsed)
+	if ( self.shouldRefresh ) then
+		L_UIDropDownMenu_RefreshDropDownSize(self);
+		self.shouldRefresh = false;
+	end
+
+	if ( not self.showTimer or not self.isCounting ) then
+		return;
+	elseif ( self.showTimer < 0 ) then
+		self:Hide();
+		self.showTimer = nil;
+		self.isCounting = nil;
+	else
+		self.showTimer = self.showTimer - elapsed;
+	end
+end
+
+-- Start the countdown on a frame
+function L_UIDropDownMenu_StartCounting(frame)
+	if ( frame.parent ) then
+		L_UIDropDownMenu_StartCounting(frame.parent);
+	else
+		frame.showTimer = L_UIDROPDOWNMENU_SHOW_TIME;
+		frame.isCounting = 1;
+	end
+end
+
+-- Stop the countdown on a frame
+function L_UIDropDownMenu_StopCounting(frame)
+	if ( frame.parent ) then
+		L_UIDropDownMenu_StopCounting(frame.parent);
+	else
+		frame.isCounting = nil;
+	end
+end
+
+function L_UIDropDownMenuButtonInvisibleButton_OnEnter(self)
+	L_UIDropDownMenu_StopCounting(self:GetParent():GetParent());
+	L_CloseDropDownMenus(self:GetParent():GetParent():GetID() + 1);
+	local parent = self:GetParent();
+	if ( parent.tooltipTitle and parent.tooltipWhileDisabled) then
+		if ( parent.tooltipOnButton ) then
+			GameTooltip:SetOwner(parent, "ANCHOR_RIGHT");
+			GameTooltip_SetTitle(GameTooltip, parent.tooltipTitle);
+			if parent.tooltipInstruction then
+				GameTooltip_AddInstructionLine(GameTooltip, parent.tooltipInstruction);
+			end
+			if parent.tooltipText then
+				GameTooltip_AddNormalLine(GameTooltip, parent.tooltipText, true);
+			end
+			if parent.tooltipWarning then
+				GameTooltip_AddColoredLine(GameTooltip, parent.tooltipWarning, RED_FONT_COLOR, true);
+			end
+			GameTooltip:Show();
+		else
+			GameTooltip_AddNewbieTip(parent, parent.tooltipTitle, 1.0, 1.0, 1.0, parent.tooltipText, 1);
+		end
+	end
+end
+
+function L_UIDropDownMenuButtonInvisibleButton_OnLeave(self)
+	L_UIDropDownMenu_StartCounting(self:GetParent():GetParent());
+	GameTooltip:Hide();
+end
+
+function L_UIDropDownMenuButton_OnEnter(self)
+	if ( self.hasArrow ) then
+		local level =  self:GetParent():GetID() + 1;
+		local listFrame = _G["L_DropDownList"..level];
+		if ( not listFrame or not listFrame:IsShown() or select(2, listFrame:GetPoint()) ~= self ) then
+			L_ToggleDropDownMenu(self:GetParent():GetID() + 1, self.value, nil, nil, nil, nil, self.menuList, self);
+		end
+	else
+		L_CloseDropDownMenus(self:GetParent():GetID() + 1);
+	end
+	self.Highlight:Show();
+	L_UIDropDownMenu_StopCounting(self:GetParent());
+	if ( self.tooltipTitle and not self.noTooltipWhileEnabled ) then
+		if ( self.tooltipOnButton ) then
+			GameTooltip:SetOwner(self, "ANCHOR_RIGHT");
+			GameTooltip_SetTitle(GameTooltip, self.tooltipTitle);
+			if self.tooltipText then
+				GameTooltip_AddNormalLine(GameTooltip, self.tooltipText, true);
+			end
+			GameTooltip:Show();
+		else
+			GameTooltip_AddNewbieTip(self, self.tooltipTitle, 1.0, 1.0, 1.0, self.tooltipText, 1);
+		end
+	end
+
+	if ( self.mouseOverIcon ~= nil ) then
+		self.Icon:SetTexture(self.mouseOverIcon);
+		self.Icon:Show();
+	end
+end
+
+function L_UIDropDownMenuButton_OnLeave(self)
+	self.Highlight:Hide();
+	L_UIDropDownMenu_StartCounting(self:GetParent());
+	GameTooltip:Hide();
+
+	if ( self.mouseOverIcon ~= nil ) then
+		if ( self.icon ~= nil ) then
+			self.Icon:SetTexture(self.icon);
+		else
+			self.Icon:Hide();
+		end
+	end
+end
+
+--[[
+List of button attributes
+======================================================
+info.text = [STRING]  --  The text of the button
+info.value = [ANYTHING]  --  The value that L_UIDROPDOWNMENU_MENU_VALUE is set to when the button is clicked
+info.func = [function()]  --  The function that is called when you click the button
+info.checked = [nil, true, function]  --  Check the button if true or function returns true
+info.isNotRadio = [nil, true]  --  Check the button uses radial image if false check box image if true
+info.isTitle = [nil, true]  --  If it's a title the button is disabled and the font color is set to yellow
+info.disabled = [nil, true]  --  Disable the button and show an invisible button that still traps the mouseover event so menu doesn't time out
+info.tooltipWhileDisabled = [nil, 1] -- Show the tooltip, even when the button is disabled.
+info.hasArrow = [nil, true]  --  Show the expand arrow for multilevel menus
+info.hasColorSwatch = [nil, true]  --  Show color swatch or not, for color selection
+info.r = [1 - 255]  --  Red color value of the color swatch
+info.g = [1 - 255]  --  Green color value of the color swatch
+info.b = [1 - 255]  --  Blue color value of the color swatch
+info.colorCode = [STRING] -- "|cAARRGGBB" embedded hex value of the button text color. Only used when button is enabled
+info.swatchFunc = [function()]  --  Function called by the color picker on color change
+info.hasOpacity = [nil, 1]  --  Show the opacity slider on the colorpicker frame
+info.opacity = [0.0 - 1.0]  --  Percentatge of the opacity, 1.0 is fully shown, 0 is transparent
+info.opacityFunc = [function()]  --  Function called by the opacity slider when you change its value
+info.cancelFunc = [function(previousValues)] -- Function called by the colorpicker when you click the cancel button (it takes the previous values as its argument)
+info.notClickable = [nil, 1]  --  Disable the button and color the font white
+info.notCheckable = [nil, 1]  --  Shrink the size of the buttons and don't display a check box
+info.owner = [Frame]  --  Dropdown frame that "owns" the current dropdownlist
+info.keepShownOnClick = [nil, 1]  --  Don't hide the dropdownlist after a button is clicked
+info.tooltipTitle = [nil, STRING] -- Title of the tooltip shown on mouseover
+info.tooltipText = [nil, STRING] -- Text of the tooltip shown on mouseover
+info.tooltipOnButton = [nil, 1] -- Show the tooltip attached to the button instead of as a Newbie tooltip.
+info.justifyH = [nil, "CENTER"] -- Justify button text
+info.arg1 = [ANYTHING] -- This is the first argument used by info.func
+info.arg2 = [ANYTHING] -- This is the second argument used by info.func
+info.fontObject = [FONT] -- font object replacement for Normal and Highlight
+info.menuTable = [TABLE] -- This contains an array of info tables to be displayed as a child menu
+info.noClickSound = [nil, 1]  --  Set to 1 to suppress the sound when clicking the button. The sound only plays if .func is set.
+info.padding = [nil, NUMBER] -- Number of pixels to pad the text on the right side
+info.leftPadding = [nil, NUMBER] -- Number of pixels to pad the button on the left side
+info.minWidth = [nil, NUMBER] -- Minimum width for this line
+info.customFrame = frame -- Allows this button to be a completely custom frame, should inherit from L_UIDropDownCustomMenuEntryTemplate and override appropriate methods.
+info.icon = [TEXTURE] -- An icon for the button.
+info.mouseOverIcon = [TEXTURE] -- An override icon when a button is moused over.
+]]
+
+local L_UIDropDownMenu_ButtonInfo = {};
+
+--Until we get around to making this betterz...
+--local UIDropDownMenu_SecureInfo = {};
+
+--local wipe = table.wipe;
+
+function L_UIDropDownMenu_CreateInfo()
+	-- Reuse the same table to prevent memory churn
+
+--	if ( issecure() ) then
+--		securecall(wipe, UIDropDownMenu_SecureInfo);
+--		return UIDropDownMenu_SecureInfo;
+--	else
+		return wipe(L_UIDropDownMenu_ButtonInfo);
+--	end
+end
+
+function L_UIDropDownMenu_CreateFrames(level, index)
+
+	while ( level > L_UIDROPDOWNMENU_MAXLEVELS ) do
+		L_UIDROPDOWNMENU_MAXLEVELS = L_UIDROPDOWNMENU_MAXLEVELS + 1;
+		local newList = CreateFrame("Button", "L_DropDownList"..L_UIDROPDOWNMENU_MAXLEVELS, nil, "L_UIDropDownListTemplate");
+		newList:SetFrameStrata("FULLSCREEN_DIALOG");
+		newList:SetToplevel(true);
+		newList:Hide();
+		newList:SetID(L_UIDROPDOWNMENU_MAXLEVELS);
+		newList:SetWidth(180)
+		newList:SetHeight(10)
+		for i=L_UIDROPDOWNMENU_MINBUTTONS+1, L_UIDROPDOWNMENU_MAXBUTTONS do
+			local newButton = CreateFrame("Button", "L_DropDownList"..L_UIDROPDOWNMENU_MAXLEVELS.."Button"..i, newList, "L_UIDropDownMenuButtonTemplate");
+			newButton:SetID(i);
+		end
+	end
+
+	while ( index > L_UIDROPDOWNMENU_MAXBUTTONS ) do
+		L_UIDROPDOWNMENU_MAXBUTTONS = L_UIDROPDOWNMENU_MAXBUTTONS + 1;
+		for i=1, L_UIDROPDOWNMENU_MAXLEVELS do
+			local newButton = CreateFrame("Button", "L_DropDownList"..i.."Button"..L_UIDROPDOWNMENU_MAXBUTTONS, _G["L_DropDownList"..i], "L_UIDropDownMenuButtonTemplate");
+			newButton:SetID(L_UIDROPDOWNMENU_MAXBUTTONS);
+		end
+	end
+end
+
+local separatorInfo;
+
+function L_UIDropDownMenu_AddSeparator(level)
+	if not separatorInfo then
+		separatorInfo =	{
+			hasArrow = false;
+			dist = 0;
+			isTitle = true;
+			isUninteractable = true;
+			notCheckable = true;
+			iconOnly = true;
+			icon = "Interface\\Common\\UI-TooltipDivider-Transparent";
+			tCoordLeft = 0;
+			tCoordRight = 1;
+			tCoordTop = 0;
+			tCoordBottom = 1;
+			tSizeX = 0;
+			tSizeY = 8;
+			tFitDropDownSizeX = true;
+			iconInfo = {
+				tCoordLeft = 0,
+				tCoordRight = 1,
+				tCoordTop = 0,
+				tCoordBottom = 1,
+				tSizeX = 0,
+				tSizeY = 8,
+				tFitDropDownSizeX = true
+			},
+		};
+	end
+
+	L_UIDropDownMenu_AddButton(separatorInfo, level);
+end
+
+function L_UIDropDownMenu_AddButton(info, level)
+	--[[
+	Might to uncomment this if there are performance issues
+	if ( not L_UIDROPDOWNMENU_OPEN_MENU ) then
+		return;
+	end
+	]]
+	if ( not level ) then
+		level = 1;
+	end
+
+	local listFrame = _G["L_DropDownList"..level];
+	local index = listFrame and (listFrame.numButtons + 1) or 1;
+	local width;
+
+	L_UIDropDownMenuDelegate:SetAttribute("createframes-level", level);
+	L_UIDropDownMenuDelegate:SetAttribute("createframes-index", index);
+	L_UIDropDownMenuDelegate:SetAttribute("createframes", true);
+
+	listFrame = listFrame or _G["L_DropDownList"..level];
+	local listFrameName = listFrame:GetName();
+
+	-- Set the number of buttons in the listframe
+	listFrame.numButtons = index;
+
+	local button = _G[listFrameName.."Button"..index];
+	local normalText = _G[button:GetName().."NormalText"];
+	local icon = _G[button:GetName().."Icon"];
+	-- This button is used to capture the mouse OnEnter/OnLeave events if the dropdown button is disabled, since a disabled button doesn't receive any events
+	-- This is used specifically for drop down menu time outs
+	local invisibleButton = _G[button:GetName().."InvisibleButton"];
+
+	-- Default settings
+	button:SetDisabledFontObject(GameFontDisableSmallLeft);
+	invisibleButton:Hide();
+	button:Enable();
+
+	-- If not clickable then disable the button and set it white
+	if ( info.notClickable ) then
+		info.disabled = true;
+		button:SetDisabledFontObject(GameFontHighlightSmallLeft);
+	end
+
+	-- Set the text color and disable it if its a title
+	if ( info.isTitle ) then
+		info.disabled = true;
+		button:SetDisabledFontObject(GameFontNormalSmallLeft);
+	end
+
+	-- Disable the button if disabled and turn off the color code
+	if ( info.disabled ) then
+		button:Disable();
+		invisibleButton:Show();
+		info.colorCode = nil;
+	end
+
+	-- If there is a color for a disabled line, set it
+	if( info.disablecolor ) then
+		info.colorCode = info.disablecolor;
+	end
+
+	-- Configure button
+	if ( info.text ) then
+		-- look for inline color code this is only if the button is enabled
+		if ( info.colorCode ) then
+			button:SetText(info.colorCode..info.text.."|r");
+		else
+			button:SetText(info.text);
+		end
+
+		-- Set icon
+		if ( info.icon or info.mouseOverIcon ) then
+			icon:SetSize(16,16);
+			icon:SetTexture(info.icon);
+			icon:ClearAllPoints();
+			icon:SetPoint("RIGHT");
+
+			if ( info.tCoordLeft ) then
+				icon:SetTexCoord(info.tCoordLeft, info.tCoordRight, info.tCoordTop, info.tCoordBottom);
+			else
+				icon:SetTexCoord(0, 1, 0, 1);
+			end
+			icon:Show();
+		else
+			icon:Hide();
+		end
+
+		-- Check to see if there is a replacement font
+		if ( info.fontObject ) then
+			button:SetNormalFontObject(info.fontObject);
+			button:SetHighlightFontObject(info.fontObject);
+		else
+			button:SetNormalFontObject(GameFontHighlightSmallLeft);
+			button:SetHighlightFontObject(GameFontHighlightSmallLeft);
+		end
+	else
+		button:SetText("");
+		icon:Hide();
+	end
+
+	button.iconOnly = nil;
+	button.icon = nil;
+	button.iconInfo = nil;
+
+	if (info.iconInfo) then
+		icon.tFitDropDownSizeX = info.iconInfo.tFitDropDownSizeX;
+	else
+		icon.tFitDropDownSizeX = nil;
+	end
+	if (info.iconOnly and info.icon) then
+		button.iconOnly = true;
+		button.icon = info.icon;
+		button.iconInfo = info.iconInfo;
+
+		L_UIDropDownMenu_SetIconImage(icon, info.icon, info.iconInfo);
+		icon:ClearAllPoints();
+		icon:SetPoint("LEFT");
+	end
+
+	-- Pass through attributes
+	button.func = info.func;
+	button.owner = info.owner;
+	button.hasOpacity = info.hasOpacity;
+	button.opacity = info.opacity;
+	button.opacityFunc = info.opacityFunc;
+	button.cancelFunc = info.cancelFunc;
+	button.swatchFunc = info.swatchFunc;
+	button.keepShownOnClick = info.keepShownOnClick;
+	button.tooltipTitle = info.tooltipTitle;
+	button.tooltipText = info.tooltipText;
+	button.tooltipInstruction = info.tooltipInstruction;
+	button.tooltipWarning = info.tooltipWarning;
+	button.arg1 = info.arg1;
+	button.arg2 = info.arg2;
+	button.hasArrow = info.hasArrow;
+	button.hasColorSwatch = info.hasColorSwatch;
+	button.notCheckable = info.notCheckable;
+	button.menuList = info.menuList;
+	button.tooltipWhileDisabled = info.tooltipWhileDisabled;
+	button.noTooltipWhileEnabled = info.noTooltipWhileEnabled;
+	button.tooltipOnButton = info.tooltipOnButton;
+	button.noClickSound = info.noClickSound;
+	button.padding = info.padding;
+	button.icon = info.icon;
+	button.mouseOverIcon = info.mouseOverIcon;
+
+	if ( info.value ) then
+		button.value = info.value;
+	elseif ( info.text ) then
+		button.value = info.text;
+	else
+		button.value = nil;
+	end
+
+	local expandArrow = _G[listFrameName.."Button"..index.."ExpandArrow"];
+	expandArrow:SetShown(info.hasArrow);
+	expandArrow:SetEnabled(not info.disabled);
+
+	-- If not checkable move everything over to the left to fill in the gap where the check would be
+	local xPos = 5;
+	local yPos = -((button:GetID() - 1) * L_UIDROPDOWNMENU_BUTTON_HEIGHT) - L_UIDROPDOWNMENU_BORDER_HEIGHT;
+	local displayInfo = normalText;
+	if (info.iconOnly) then
+		displayInfo = icon;
+	end
+
+	displayInfo:ClearAllPoints();
+	if ( info.notCheckable ) then
+		if ( info.justifyH and info.justifyH == "CENTER" ) then
+			displayInfo:SetPoint("CENTER", button, "CENTER", -7, 0);
+		else
+			displayInfo:SetPoint("LEFT", button, "LEFT", 0, 0);
+		end
+		xPos = xPos + 10;
+
+	else
+		xPos = xPos + 12;
+		displayInfo:SetPoint("LEFT", button, "LEFT", 20, 0);
+	end
+
+	-- Adjust offset if displayMode is menu
+	local frame = L_UIDROPDOWNMENU_OPEN_MENU;
+	if ( frame and frame.displayMode == "MENU" ) then
+		if ( not info.notCheckable ) then
+			xPos = xPos - 6;
+		end
+	end
+
+	-- If no open frame then set the frame to the currently initialized frame
+	frame = frame or L_UIDROPDOWNMENU_INIT_MENU;
+
+	if ( info.leftPadding ) then
+		xPos = xPos + info.leftPadding;
+	end
+	button:SetPoint("TOPLEFT", button:GetParent(), "TOPLEFT", xPos, yPos);
+
+	-- See if button is selected by id or name
+	if ( frame ) then
+		if ( L_UIDropDownMenu_GetSelectedName(frame) ) then
+			if ( button:GetText() == L_UIDropDownMenu_GetSelectedName(frame) ) then
+				info.checked = 1;
+			end
+		elseif ( L_UIDropDownMenu_GetSelectedID(frame) ) then
+			if ( button:GetID() == L_UIDropDownMenu_GetSelectedID(frame) ) then
+				info.checked = 1;
+			end
+		elseif ( L_UIDropDownMenu_GetSelectedValue(frame) ) then
+			if ( button.value == L_UIDropDownMenu_GetSelectedValue(frame) ) then
+				info.checked = 1;
+			end
+		end
+	end
+
+	if not info.notCheckable then
+		local check = _G[listFrameName.."Button"..index.."Check"];
+		local uncheck = _G[listFrameName.."Button"..index.."UnCheck"];
+		if ( info.disabled ) then
+			check:SetDesaturated(true);
+			check:SetAlpha(0.5);
+			uncheck:SetDesaturated(true);
+			uncheck:SetAlpha(0.5);
+		else
+			check:SetDesaturated(false);
+			check:SetAlpha(1);
+			uncheck:SetDesaturated(false);
+			uncheck:SetAlpha(1);
+		end
+
+		if info.customCheckIconAtlas or info.customCheckIconTexture then
+			check:SetTexCoord(0, 1, 0, 1);
+			uncheck:SetTexCoord(0, 1, 0, 1);
+
+			if info.customCheckIconAtlas then
+				check:SetAtlas(info.customCheckIconAtlas);
+				uncheck:SetAtlas(info.customUncheckIconAtlas or info.customCheckIconAtlas);
+			else
+				check:SetTexture(info.customCheckIconTexture);
+				uncheck:SetTexture(info.customUncheckIconTexture or info.customCheckIconTexture);
+			end
+		elseif info.isNotRadio then
+			check:SetTexCoord(0.0, 0.5, 0.0, 0.5);
+			check:SetTexture("Interface\\Common\\UI-DropDownRadioChecks");
+			uncheck:SetTexCoord(0.5, 1.0, 0.0, 0.5);
+			uncheck:SetTexture("Interface\\Common\\UI-DropDownRadioChecks");
+		else
+			check:SetTexCoord(0.0, 0.5, 0.5, 1.0);
+			check:SetTexture("Interface\\Common\\UI-DropDownRadioChecks");
+			uncheck:SetTexCoord(0.5, 1.0, 0.5, 1.0);
+			uncheck:SetTexture("Interface\\Common\\UI-DropDownRadioChecks");
+		end
+
+		-- Checked can be a function now
+		local checked = info.checked;
+		if ( type(checked) == "function" ) then
+			checked = checked(button);
+		end
+
+		-- Show the check if checked
+		if ( checked ) then
+			button:LockHighlight();
+			check:Show();
+			uncheck:Hide();
+		else
+			button:UnlockHighlight();
+			check:Hide();
+			uncheck:Show();
+		end
+	else
+		_G[listFrameName.."Button"..index.."Check"]:Hide();
+		_G[listFrameName.."Button"..index.."UnCheck"]:Hide();
+	end
+	button.checked = info.checked;
+
+	-- If has a colorswatch, show it and vertex color it
+	local colorSwatch = _G[listFrameName.."Button"..index.."ColorSwatch"];
+	if ( info.hasColorSwatch ) then
+		_G["L_DropDownList"..level.."Button"..index.."ColorSwatch".."NormalTexture"]:SetVertexColor(info.r, info.g, info.b);
+		button.r = info.r;
+		button.g = info.g;
+		button.b = info.b;
+		colorSwatch:Show();
+	else
+		colorSwatch:Hide();
+	end
+
+	L_UIDropDownMenu_CheckAddCustomFrame(listFrame, button, info);
+
+	button:SetShown(button.customFrame == nil);
+
+	button.minWidth = info.minWidth;
+
+	width = max(L_UIDropDownMenu_GetButtonWidth(button), info.minWidth or 0);
+	--Set maximum button width
+	if ( width > listFrame.maxWidth ) then
+		listFrame.maxWidth = width;
+	end
+
+	-- Set the height of the listframe
+	listFrame:SetHeight((index * L_UIDROPDOWNMENU_BUTTON_HEIGHT) + (L_UIDROPDOWNMENU_BORDER_HEIGHT * 2));
+end
+
+function L_UIDropDownMenu_CheckAddCustomFrame(self, button, info)
+	local customFrame = info.customFrame;
+	button.customFrame = customFrame;
+	if customFrame then
+		customFrame:SetOwningButton(button);
+		customFrame:ClearAllPoints();
+		customFrame:SetPoint("TOPLEFT", button, "TOPLEFT", 0, 0);
+		customFrame:Show();
+
+		L_UIDropDownMenu_RegisterCustomFrame(self, customFrame);
+	end
+end
+
+function L_UIDropDownMenu_RegisterCustomFrame(self, customFrame)
+	self.customFrames = self.customFrames or {}
+	table.insert(self.customFrames, customFrame);
+end
+
+function L_UIDropDownMenu_GetMaxButtonWidth(self)
+	local maxWidth = 0;
+	for i=1, self.numButtons do
+		local button = _G[self:GetName().."Button"..i];
+		local width = L_UIDropDownMenu_GetButtonWidth(button);
+		if ( width > maxWidth ) then
+			maxWidth = width;
+		end
+	end
+	return maxWidth;
+end
+
+function L_UIDropDownMenu_GetButtonWidth(button)
+	local minWidth = button.minWidth or 0;
+	if button.customFrame and button.customFrame:IsShown() then
+		return math.max(minWidth, button.customFrame:GetPreferredEntryWidth());
+	end
+
+	if not button:IsShown() then
+		return 0;
+	end
+
+	local width;
+	local buttonName = button:GetName();
+	local icon = _G[buttonName.."Icon"];
+	local normalText = _G[buttonName.."NormalText"];
+
+	if ( button.iconOnly and icon ) then
+		width = icon:GetWidth();
+	elseif ( normalText and normalText:GetText() ) then
+		width = normalText:GetWidth() + 40;
+
+		if ( button.icon ) then
+			-- Add padding for the icon
+			width = width + 10;
+		end
+	else
+		return minWidth;
+	end
+
+	-- Add padding if has and expand arrow or color swatch
+	if ( button.hasArrow or button.hasColorSwatch ) then
+		width = width + 10;
+	end
+	if ( button.notCheckable ) then
+		width = width - 30;
+	end
+	if ( button.padding ) then
+		width = width + button.padding;
+	end
+
+	return math.max(minWidth, width);
+end
+
+function L_UIDropDownMenu_Refresh(frame, useValue, dropdownLevel)
+	local button, checked, checkImage, uncheckImage, normalText, width;
+	local maxWidth = 0;
+	local somethingChecked = nil;
+	if ( not dropdownLevel ) then
+		dropdownLevel = L_UIDROPDOWNMENU_MENU_LEVEL;
+	end
+
+	local listFrame = _G["L_DropDownList"..dropdownLevel];
+	listFrame.numButtons = listFrame.numButtons or 0;
+	-- Just redraws the existing menu
+	for i=1, L_UIDROPDOWNMENU_MAXBUTTONS do
+		button = _G["L_DropDownList"..dropdownLevel.."Button"..i];
+		checked = nil;
+
+		if(i <= listFrame.numButtons) then
+			-- See if checked or not
+			if ( L_UIDropDownMenu_GetSelectedName(frame) ) then
+				if ( button:GetText() == L_UIDropDownMenu_GetSelectedName(frame) ) then
+					checked = 1;
+				end
+			elseif ( L_UIDropDownMenu_GetSelectedID(frame) ) then
+				if ( button:GetID() == L_UIDropDownMenu_GetSelectedID(frame) ) then
+					checked = 1;
+				end
+			elseif ( L_UIDropDownMenu_GetSelectedValue(frame) ) then
+				if ( button.value == L_UIDropDownMenu_GetSelectedValue(frame) ) then
+					checked = 1;
+				end
+			end
+		end
+		if (button.checked and type(button.checked) == "function") then
+			checked = button.checked(button);
+		end
+
+		if not button.notCheckable and button:IsShown() then
+			-- If checked show check image
+			checkImage = _G["L_DropDownList"..dropdownLevel.."Button"..i.."Check"];
+			uncheckImage = _G["L_DropDownList"..dropdownLevel.."Button"..i.."UnCheck"];
+			if ( checked ) then
+				somethingChecked = true;
+				local icon = GetChild(frame, frame:GetName(), "Icon");
+				if (button.iconOnly and icon and button.icon) then
+					L_UIDropDownMenu_SetIconImage(icon, button.icon, button.iconInfo);
+				elseif ( useValue ) then
+					L_UIDropDownMenu_SetText(frame, button.value);
+					icon:Hide();
+				else
+					L_UIDropDownMenu_SetText(frame, button:GetText());
+					icon:Hide();
+				end
+				button:LockHighlight();
+				checkImage:Show();
+				uncheckImage:Hide();
+			else
+				button:UnlockHighlight();
+				checkImage:Hide();
+				uncheckImage:Show();
+			end
+		end
+
+		if ( button:IsShown() ) then
+			width = L_UIDropDownMenu_GetButtonWidth(button);
+			if ( width > maxWidth ) then
+				maxWidth = width;
+			end
+		end
+	end
+	if(somethingChecked == nil) then
+		L_UIDropDownMenu_SetText(frame, VIDEO_QUALITY_LABEL6);
+	end
+	if (not frame.noResize) then
+		for i=1, L_UIDROPDOWNMENU_MAXBUTTONS do
+			button = _G["L_DropDownList"..dropdownLevel.."Button"..i];
+			button:SetWidth(maxWidth);
+		end
+		L_UIDropDownMenu_RefreshDropDownSize(_G["L_DropDownList"..dropdownLevel]);
+	end
+end
+
+function L_UIDropDownMenu_RefreshAll(frame, useValue)
+	for dropdownLevel = L_UIDROPDOWNMENU_MENU_LEVEL, 2, -1 do
+		local listFrame = _G["L_DropDownList"..dropdownLevel];
+		if ( listFrame:IsShown() ) then
+			L_UIDropDownMenu_Refresh(frame, nil, dropdownLevel);
+		end
+	end
+	-- useValue is the text on the dropdown, only needs to be set once
+	L_UIDropDownMenu_Refresh(frame, useValue, 1);
+end
+
+function L_UIDropDownMenu_SetIconImage(icon, texture, info)
+	icon:SetTexture(texture);
+	if ( info.tCoordLeft ) then
+		icon:SetTexCoord(info.tCoordLeft, info.tCoordRight, info.tCoordTop, info.tCoordBottom);
+	else
+		icon:SetTexCoord(0, 1, 0, 1);
+	end
+	if ( info.tSizeX ) then
+		icon:SetWidth(info.tSizeX);
+	else
+		icon:SetWidth(16);
+	end
+	if ( info.tSizeY ) then
+		icon:SetHeight(info.tSizeY);
+	else
+		icon:SetHeight(16);
+	end
+	icon:Show();
+end
+
+function L_UIDropDownMenu_SetSelectedName(frame, name, useValue)
+	frame.selectedName = name;
+	frame.selectedID = nil;
+	frame.selectedValue = nil;
+	L_UIDropDownMenu_Refresh(frame, useValue);
+end
+
+function L_UIDropDownMenu_SetSelectedValue(frame, value, useValue)
+	-- useValue will set the value as the text, not the name
+	frame.selectedName = nil;
+	frame.selectedID = nil;
+	frame.selectedValue = value;
+	L_UIDropDownMenu_Refresh(frame, useValue);
+end
+
+function L_UIDropDownMenu_SetSelectedID(frame, id, useValue)
+	frame.selectedID = id;
+	frame.selectedName = nil;
+	frame.selectedValue = nil;
+	L_UIDropDownMenu_Refresh(frame, useValue);
+end
+
+function L_UIDropDownMenu_GetSelectedName(frame)
+	return frame.selectedName;
+end
+
+function L_UIDropDownMenu_GetSelectedID(frame)
+	if ( frame.selectedID ) then
+		return frame.selectedID;
+	else
+		-- If no explicit selectedID then try to send the id of a selected value or name
+		local button;
+		for i=1, L_UIDROPDOWNMENU_MAXBUTTONS do
+			button = _G["L_DropDownList"..L_UIDROPDOWNMENU_MENU_LEVEL.."Button"..i];
+			-- See if checked or not
+			if ( L_UIDropDownMenu_GetSelectedName(frame) ) then
+				if ( button:GetText() == L_UIDropDownMenu_GetSelectedName(frame) ) then
+					return i;
+				end
+			elseif ( L_UIDropDownMenu_GetSelectedValue(frame) ) then
+				if ( button.value == L_UIDropDownMenu_GetSelectedValue(frame) ) then
+					return i;
+				end
+			end
+		end
+	end
+end
+
+function L_UIDropDownMenu_GetSelectedValue(frame)
+	return frame.selectedValue;
+end
+
+function L_UIDropDownMenuButton_OnClick(self)
+	local checked = self.checked;
+	if ( type (checked) == "function" ) then
+		checked = checked(self);
+	end
+
+
+	if ( self.keepShownOnClick ) then
+		if not self.notCheckable then
+			if ( checked ) then
+				_G[self:GetName().."Check"]:Hide();
+				_G[self:GetName().."UnCheck"]:Show();
+				checked = false;
+			else
+				_G[self:GetName().."Check"]:Show();
+				_G[self:GetName().."UnCheck"]:Hide();
+				checked = true;
+			end
+		end
+	else
+		self:GetParent():Hide();
+	end
+
+	if ( type (self.checked) ~= "function" ) then
+		self.checked = checked;
+	end
+
+	-- saving this here because func might use a dropdown, changing this self's attributes
+	local playSound = true;
+	if ( self.noClickSound ) then
+		playSound = false;
+	end
+
+	local func = self.func;
+	if ( func ) then
+		func(self, self.arg1, self.arg2, checked);
+	else
+		return;
+	end
+
+	if ( playSound ) then
+		PlaySound(SOUNDKIT.U_CHAT_SCROLL_BUTTON);
+	end
+end
+
+function L_HideDropDownMenu(level)
+	local listFrame = _G["L_DropDownList"..level];
+	listFrame:Hide();
+end
+
+function L_ToggleDropDownMenu(level, value, dropDownFrame, anchorName, xOffset, yOffset, menuList, button, autoHideDelay)
+	if ( not level ) then
+		level = 1;
+	end
+	L_UIDropDownMenuDelegate:SetAttribute("createframes-level", level);
+	L_UIDropDownMenuDelegate:SetAttribute("createframes-index", 0);
+	L_UIDropDownMenuDelegate:SetAttribute("createframes", true);
+	L_UIDROPDOWNMENU_MENU_LEVEL = level;
+	L_UIDROPDOWNMENU_MENU_VALUE = value;
+	local listFrame = _G["L_DropDownList"..level];
+	local listFrameName = "L_DropDownList"..level;
+	local tempFrame;
+	local point, relativePoint, relativeTo;
+	if ( not dropDownFrame ) then
+		tempFrame = button:GetParent();
+	else
+		tempFrame = dropDownFrame;
+	end
+	if ( listFrame:IsShown() and (L_UIDROPDOWNMENU_OPEN_MENU == tempFrame) ) then
+		listFrame:Hide();
+	else
+		-- Set the dropdownframe scale
+		local uiScale;
+		local uiParentScale = UIParent:GetScale();
+		if ( GetCVar("useUIScale") == "1" ) then
+			uiScale = tonumber(GetCVar("uiscale"));
+			if ( uiParentScale < uiScale ) then
+				uiScale = uiParentScale;
+			end
+		else
+			uiScale = uiParentScale;
+		end
+		listFrame:SetScale(uiScale);
+
+		-- Hide the listframe anyways since it is redrawn OnShow()
+		listFrame:Hide();
+
+		-- Frame to anchor the dropdown menu to
+		local anchorFrame;
+
+		-- Display stuff
+		-- Level specific stuff
+		if ( level == 1 ) then
+			L_UIDropDownMenuDelegate:SetAttribute("openmenu", dropDownFrame);
+			listFrame:ClearAllPoints();
+			-- If there's no specified anchorName then use left side of the dropdown menu
+			if ( not anchorName ) then
+				-- See if the anchor was set manually using setanchor
+				if ( dropDownFrame.xOffset ) then
+					xOffset = dropDownFrame.xOffset;
+				end
+				if ( dropDownFrame.yOffset ) then
+					yOffset = dropDownFrame.yOffset;
+				end
+				if ( dropDownFrame.point ) then
+					point = dropDownFrame.point;
+				end
+				if ( dropDownFrame.relativeTo ) then
+					relativeTo = dropDownFrame.relativeTo;
+				else
+					relativeTo = GetChild(L_UIDROPDOWNMENU_OPEN_MENU, L_UIDROPDOWNMENU_OPEN_MENU:GetName(), "Left");
+				end
+				if ( dropDownFrame.relativePoint ) then
+					relativePoint = dropDownFrame.relativePoint;
+				end
+			elseif ( anchorName == "cursor" ) then
+				relativeTo = nil;
+				local cursorX, cursorY = GetCursorPosition();
+				cursorX = cursorX/uiScale;
+				cursorY =  cursorY/uiScale;
+
+				if ( not xOffset ) then
+					xOffset = 0;
+				end
+				if ( not yOffset ) then
+					yOffset = 0;
+				end
+				xOffset = cursorX + xOffset;
+				yOffset = cursorY + yOffset;
+			else
+				-- See if the anchor was set manually using setanchor
+				if ( dropDownFrame.xOffset ) then
+					xOffset = dropDownFrame.xOffset;
+				end
+				if ( dropDownFrame.yOffset ) then
+					yOffset = dropDownFrame.yOffset;
+				end
+				if ( dropDownFrame.point ) then
+					point = dropDownFrame.point;
+				end
+				if ( dropDownFrame.relativeTo ) then
+					relativeTo = dropDownFrame.relativeTo;
+				else
+					relativeTo = anchorName;
+				end
+				if ( dropDownFrame.relativePoint ) then
+					relativePoint = dropDownFrame.relativePoint;
+				end
+			end
+			if ( not xOffset or not yOffset ) then
+				xOffset = 8;
+				yOffset = 22;
+			end
+			if ( not point ) then
+				point = "TOPLEFT";
+			end
+			if ( not relativePoint ) then
+				relativePoint = "BOTTOMLEFT";
+			end
+			listFrame:SetPoint(point, relativeTo, relativePoint, xOffset, yOffset);
+		else
+			if ( not dropDownFrame ) then
+				dropDownFrame = L_UIDROPDOWNMENU_OPEN_MENU;
+			end
+			listFrame:ClearAllPoints();
+			-- If this is a dropdown button, not the arrow anchor it to itself
+			if ( strsub(button:GetParent():GetName(), 0,14) == "L_DropDownList" and strlen(button:GetParent():GetName()) == 15 ) then
+				anchorFrame = button;
+			else
+				anchorFrame = button:GetParent();
+			end
+			point = "TOPLEFT";
+			relativePoint = "TOPRIGHT";
+			listFrame:SetPoint(point, anchorFrame, relativePoint, 0, 0);
+		end
+
+		-- Change list box appearance depending on display mode
+		if ( dropDownFrame and dropDownFrame.displayMode == "MENU" ) then
+			_G[listFrameName.."Backdrop"]:Hide();
+			_G[listFrameName.."MenuBackdrop"]:Show();
+		else
+			_G[listFrameName.."Backdrop"]:Show();
+			_G[listFrameName.."MenuBackdrop"]:Hide();
+		end
+		dropDownFrame.menuList = menuList;
+		L_UIDropDownMenu_Initialize(dropDownFrame, dropDownFrame.initialize, nil, level, menuList);
+		-- If no items in the drop down don't show it
+		if ( listFrame.numButtons == 0 ) then
+			return;
+		end
+
+		-- Check to see if the dropdownlist is off the screen, if it is anchor it to the top of the dropdown button
+		listFrame:Show();
+		-- Hack since GetCenter() is returning coords relative to 1024x768
+		local x, y = listFrame:GetCenter();
+		-- Hack will fix this in next revision of dropdowns
+		if ( not x or not y ) then
+			listFrame:Hide();
+			return;
+		end
+
+		listFrame.onHide = dropDownFrame.onHide;
+
+
+		--  We just move level 1 enough to keep it on the screen. We don't necessarily change the anchors.
+		if ( level == 1 ) then
+			local offLeft = listFrame:GetLeft()/uiScale;
+			local offRight = (GetScreenWidth() - listFrame:GetRight())/uiScale;
+			local offTop = (GetScreenHeight() - listFrame:GetTop())/uiScale;
+			local offBottom = listFrame:GetBottom()/uiScale;
+
+			local xAddOffset, yAddOffset = 0, 0;
+			if ( offLeft < 0 ) then
+				xAddOffset = -offLeft;
+			elseif ( offRight < 0 ) then
+				xAddOffset = offRight;
+			end
+
+			if ( offTop < 0 ) then
+				yAddOffset = offTop;
+			elseif ( offBottom < 0 ) then
+				yAddOffset = -offBottom;
+			end
+
+			listFrame:ClearAllPoints();
+			if ( anchorName == "cursor" ) then
+				listFrame:SetPoint(point, relativeTo, relativePoint, xOffset + xAddOffset, yOffset + yAddOffset);
+			else
+				listFrame:SetPoint(point, relativeTo, relativePoint, xOffset + xAddOffset, yOffset + yAddOffset);
+			end
+		else
+			-- Determine whether the menu is off the screen or not
+			local offscreenY, offscreenX;
+			if ( (y - listFrame:GetHeight()/2) < 0 ) then
+				offscreenY = 1;
+			end
+			if ( listFrame:GetRight() > GetScreenWidth() ) then
+				offscreenX = 1;
+			end
+			if ( offscreenY and offscreenX ) then
+				point = gsub(point, "TOP(.*)", "BOTTOM%1");
+				point = gsub(point, "(.*)LEFT", "%1RIGHT");
+				relativePoint = gsub(relativePoint, "TOP(.*)", "BOTTOM%1");
+				relativePoint = gsub(relativePoint, "(.*)RIGHT", "%1LEFT");
+				xOffset = -11;
+				yOffset = -14;
+			elseif ( offscreenY ) then
+				point = gsub(point, "TOP(.*)", "BOTTOM%1");
+				relativePoint = gsub(relativePoint, "TOP(.*)", "BOTTOM%1");
+				xOffset = 0;
+				yOffset = -14;
+			elseif ( offscreenX ) then
+				point = gsub(point, "(.*)LEFT", "%1RIGHT");
+				relativePoint = gsub(relativePoint, "(.*)RIGHT", "%1LEFT");
+				xOffset = -11;
+				yOffset = 14;
+			else
+				xOffset = 0;
+				yOffset = 14;
+			end
+
+			listFrame:ClearAllPoints();
+			listFrame.parentLevel = tonumber(strmatch(anchorFrame:GetName(), "L_DropDownList(%d+)"));
+			listFrame.parentID = anchorFrame:GetID();
+			listFrame:SetPoint(point, anchorFrame, relativePoint, xOffset, yOffset);
+		end
+
+		if ( autoHideDelay and tonumber(autoHideDelay)) then
+			listFrame.showTimer = autoHideDelay;
+			listFrame.isCounting = 1;
+		end
+	end
+end
+
+function L_CloseDropDownMenus(level)
+	if ( not level ) then
+		level = 1;
+	end
+	for i=level, L_UIDROPDOWNMENU_MAXLEVELS do
+		_G["L_DropDownList"..i]:Hide();
+	end
+end
+
+function L_UIDropDownMenu_OnHide(self)
+	local id = self:GetID()
+	if ( self.onHide ) then
+		self.onHide(id+1);
+		self.onHide = nil;
+	end
+	L_CloseDropDownMenus(id+1);
+	L_OPEN_DROPDOWNMENUS[id] = nil;
+	if (id == 1) then
+		L_UIDROPDOWNMENU_OPEN_MENU = nil;
+	end
+
+	if self.customFrames then
+		for index, frame in ipairs(self.customFrames) do
+			frame:Hide();
+		end
+
+		self.customFrames = nil;
+	end
+end
+
+function L_UIDropDownMenu_SetWidth(frame, width, padding)
+	local frameName = frame:GetName();
+	GetChild(frame, frameName, "Middle"):SetWidth(width);
+	local defaultPadding = 25;
+	if ( padding ) then
+		frame:SetWidth(width + padding);
+	else
+		frame:SetWidth(width + defaultPadding + defaultPadding);
+	end
+	if ( padding ) then
+		GetChild(frame, frameName, "Text"):SetWidth(width);
+	else
+		GetChild(frame, frameName, "Text"):SetWidth(width - defaultPadding);
+	end
+	frame.noResize = 1;
+end
+
+function L_UIDropDownMenu_SetButtonWidth(frame, width)
+	local frameName = frame:GetName();
+	if ( width == "TEXT" ) then
+		width = GetChild(frame, frameName, "Text"):GetWidth();
+	end
+
+	GetChild(frame, frameName, "Button"):SetWidth(width);
+	frame.noResize = 1;
+end
+
+function L_UIDropDownMenu_SetText(frame, text)
+	local frameName = frame:GetName();
+	GetChild(frame, frameName, "Text"):SetText(text);
+end
+
+function L_UIDropDownMenu_GetText(frame)
+	local frameName = frame:GetName();
+	return GetChild(frame, frameName, "Text"):GetText();
+end
+
+function L_UIDropDownMenu_ClearAll(frame)
+	-- Previous code refreshed the menu quite often and was a performance bottleneck
+	frame.selectedID = nil;
+	frame.selectedName = nil;
+	frame.selectedValue = nil;
+	L_UIDropDownMenu_SetText(frame, "");
+
+	local button, checkImage, uncheckImage;
+	for i=1, L_UIDROPDOWNMENU_MAXBUTTONS do
+		button = _G["L_DropDownList"..L_UIDROPDOWNMENU_MENU_LEVEL.."Button"..i];
+		button:UnlockHighlight();
+
+		checkImage = _G["L_DropDownList"..L_UIDROPDOWNMENU_MENU_LEVEL.."Button"..i.."Check"];
+		checkImage:Hide();
+		uncheckImage = _G["L_DropDownList"..L_UIDROPDOWNMENU_MENU_LEVEL.."Button"..i.."UnCheck"];
+		uncheckImage:Hide();
+	end
+end
+
+function L_UIDropDownMenu_JustifyText(frame, justification)
+	local frameName = frame:GetName();
+	local text = GetChild(frame, frameName, "Text");
+	text:ClearAllPoints();
+	if ( justification == "LEFT" ) then
+		text:SetPoint("LEFT", GetChild(frame, frameName, "Left"), "LEFT", 27, 2);
+		text:SetJustifyH("LEFT");
+	elseif ( justification == "RIGHT" ) then
+		text:SetPoint("RIGHT", GetChild(frame, frameName, "Right"), "RIGHT", -43, 2);
+		text:SetJustifyH("RIGHT");
+	elseif ( justification == "CENTER" ) then
+		text:SetPoint("CENTER", GetChild(frame, frameName, "Middle"), "CENTER", -5, 2);
+		text:SetJustifyH("CENTER");
+	end
+end
+
+function L_UIDropDownMenu_SetAnchor(dropdown, xOffset, yOffset, point, relativeTo, relativePoint)
+	dropdown.xOffset = xOffset;
+	dropdown.yOffset = yOffset;
+	dropdown.point = point;
+	dropdown.relativeTo = relativeTo;
+	dropdown.relativePoint = relativePoint;
+end
+
+function L_UIDropDownMenu_GetCurrentDropDown()
+	if ( L_UIDROPDOWNMENU_OPEN_MENU ) then
+		return L_UIDROPDOWNMENU_OPEN_MENU;
+	elseif ( L_UIDROPDOWNMENU_INIT_MENU ) then
+		return L_UIDROPDOWNMENU_INIT_MENU;
+	end
+end
+
+function L_UIDropDownMenuButton_GetChecked(self)
+	return _G[self:GetName().."Check"]:IsShown();
+end
+
+function L_UIDropDownMenuButton_GetName(self)
+	return _G[self:GetName().."NormalText"]:GetText();
+end
+
+function L_UIDropDownMenuButton_OpenColorPicker(self, button)
+	securecall("CloseMenus");
+	if ( not button ) then
+		button = self;
+	end
+	L_UIDROPDOWNMENU_MENU_VALUE = button.value;
+	OpenColorPicker(button); --remains shared through color picker frame
+end
+
+function L_UIDropDownMenu_DisableButton(level, id)
+	_G["L_DropDownList"..level.."Button"..id]:Disable();
+end
+
+function L_UIDropDownMenu_EnableButton(level, id)
+	_G["L_DropDownList"..level.."Button"..id]:Enable();
+end
+
+function L_UIDropDownMenu_SetButtonText(level, id, text, colorCode)
+	local button = _G["L_DropDownList"..level.."Button"..id];
+	if ( colorCode) then
+		button:SetText(colorCode..text.."|r");
+	else
+		button:SetText(text);
+	end
+end
+
+function L_UIDropDownMenu_SetButtonNotClickable(level, id)
+	_G["L_DropDownList"..level.."Button"..id]:SetDisabledFontObject(GameFontHighlightSmallLeft);
+end
+
+function L_UIDropDownMenu_SetButtonClickable(level, id)
+	_G["L_DropDownList"..level.."Button"..id]:SetDisabledFontObject(GameFontDisableSmallLeft);
+end
+
+function L_UIDropDownMenu_DisableDropDown(dropDown)
+	local dropDownName = dropDown:GetName();
+	local label = GetChild(dropDown, dropDownName, "Label");
+	if ( label ) then
+		label:SetVertexColor(GRAY_FONT_COLOR.r, GRAY_FONT_COLOR.g, GRAY_FONT_COLOR.b);
+	end
+	GetChild(dropDown, dropDownName, "Text"):SetVertexColor(GRAY_FONT_COLOR.r, GRAY_FONT_COLOR.g, GRAY_FONT_COLOR.b);
+	GetChild(dropDown, dropDownName, "Button"):Disable();
+	dropDown.isDisabled = 1;
+end
+
+function L_UIDropDownMenu_EnableDropDown(dropDown)
+	local dropDownName = dropDown:GetName();
+	local label = GetChild(dropDown, dropDownName, "Label");
+	if ( label ) then
+		label:SetVertexColor(NORMAL_FONT_COLOR.r, NORMAL_FONT_COLOR.g, NORMAL_FONT_COLOR.b);
+	end
+	GetChild(dropDown, dropDownName, "Text"):SetVertexColor(HIGHLIGHT_FONT_COLOR.r, HIGHLIGHT_FONT_COLOR.g, HIGHLIGHT_FONT_COLOR.b);
+	GetChild(dropDown, dropDownName, "Button"):Enable();
+	dropDown.isDisabled = nil;
+end
+
+function L_UIDropDownMenu_IsEnabled(dropDown)
+	return not dropDown.isDisabled;
+end
+
+function L_UIDropDownMenu_GetValue(id)
+	--Only works if the dropdown has just been initialized, lame, I know =(
+	local button = _G["L_DropDownList1Button"..id];
+	if ( button ) then
+		return _G["L_DropDownList1Button"..id].value;
+	else
+		return nil;
+	end
+end
+
+--[[function OpenColorPicker(info) --ColorPicker stuff not changed
+	ColorPickerFrame.func = info.swatchFunc;
+	ColorPickerFrame.hasOpacity = info.hasOpacity;
+	ColorPickerFrame.opacityFunc = info.opacityFunc;
+	ColorPickerFrame.opacity = info.opacity;
+	ColorPickerFrame.previousValues = {r = info.r, g = info.g, b = info.b, opacity = info.opacity};
+	ColorPickerFrame.cancelFunc = info.cancelFunc;
+	ColorPickerFrame.extraInfo = info.extraInfo;
+	-- This must come last, since it triggers a call to ColorPickerFrame.func()
+	ColorPickerFrame:SetColorRGB(info.r, info.g, info.b);
+	ShowUIPanel(ColorPickerFrame);
+end
+
+function ColorPicker_GetPreviousValues()
+	return ColorPickerFrame.previousValues.r, ColorPickerFrame.previousValues.g, ColorPickerFrame.previousValues.b;
+end]]
diff --git a/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.xml b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.xml
new file mode 100755
index 0000000..13a3944
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenu.xml
@@ -0,0 +1,23 @@
+<!-- $Id: LibUIDropDownMenu.xml 13 2017-05-25 04:26:27Z arith $ -->
+<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="LibUIDropDownMenu.lua"/>
+	<Include file="LibUIDropDownMenuTemplates.xml"/>
+	<Script file="LibEasyMenu.lua"/>
+	<Button name="L_DropDownList1" toplevel="true" frameStrata="FULLSCREEN_DIALOG" inherits="L_UIDropDownListTemplate" hidden="true" id="1">
+		<Size>
+			<AbsDimension x="180" y="10"/>
+		</Size>
+		<Scripts>
+			<OnLoad>
+				local fontName, fontHeight, fontFlags = _G["L_DropDownList1Button1NormalText"]:GetFont();
+				L_UIDROPDOWNMENU_DEFAULT_TEXT_HEIGHT = fontHeight;
+			</OnLoad>
+		</Scripts>
+	</Button>
+	<Button name="L_DropDownList2" toplevel="true" frameStrata="FULLSCREEN_DIALOG" inherits="L_UIDropDownListTemplate" hidden="true" id="2">
+		<Size>
+			<AbsDimension x="180" y="10"/>
+		</Size>
+	</Button>
+</Ui>
diff --git a/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.lua b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.lua
new file mode 100755
index 0000000..ca74b3d
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.lua
@@ -0,0 +1,57 @@
+-- $Id: LibUIDropDownMenuTemplates.lua 30 2018-04-24 06:44:39Z arith $
+-- ----------------------------------------------------------------------------
+-- Localized Lua globals.
+-- ----------------------------------------------------------------------------
+local _G = getfenv(0)
+-- ----------------------------------------------------------------------------
+local MAJOR_VERSION = "LibUIDropDownMenuTemplates"
+local MINOR_VERSION = 90000 + tonumber(("$Rev: 30 $"):match("%d+"))
+
+local LibStub = _G.LibStub
+if not LibStub then error(MAJOR_VERSION .. " requires LibStub.") end
+local Lib = LibStub:NewLibrary(MAJOR_VERSION, MINOR_VERSION)
+if not Lib then return end
+
+-- Custom dropdown buttons are instantiated by some external system.
+-- When calling L_UIDropDownMenu_AddButton that system sets info.customFrame to the instance of the frame it wants to place on the menu.
+-- The dropdown menu creates its button for the entry as it normally would, but hides all elements.  The custom frame is then anchored
+-- to that button and assumes responsibility for all relevant dropdown menu operations.
+-- The hidden button will request a size that it should become from the custom frame.
+
+
+L_UIDropDownCustomMenuEntryMixin = {};
+
+function L_UIDropDownCustomMenuEntryMixin:GetPreferredEntryWidth()
+	-- NOTE: Only width is currently supported, dropdown menus size vertically based on how many buttons are present.
+	return self:GetWidth();
+end
+
+function L_UIDropDownCustomMenuEntryMixin:OnSetOwningButton()
+	-- for derived objects to implement
+end
+
+function L_UIDropDownCustomMenuEntryMixin:SetOwningButton(button)
+	self:SetParent(button:GetParent());
+	self.owningButton = button;
+	self:OnSetOwningButton();
+end
+
+function L_UIDropDownCustomMenuEntryMixin:GetOwningDropdown()
+	return self.owningButton:GetParent();
+end
+
+function L_UIDropDownCustomMenuEntryMixin:SetContextData(contextData)
+	self.contextData = contextData;
+end
+
+function L_UIDropDownCustomMenuEntryMixin:GetContextData()
+	return self.contextData;
+end
+
+function L_UIDropDownCustomMenuEntryMixin:OnEnter()
+	L_UIDropDownMenu_StopCounting(self:GetOwningDropdown());
+end
+
+function L_UIDropDownCustomMenuEntryMixin:OnLeave()
+	L_UIDropDownMenu_StartCounting(self:GetOwningDropdown());
+end
diff --git a/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.xml b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.xml
new file mode 100755
index 0000000..2ab55e6
--- /dev/null
+++ b/Titan/libs/!LibUIDropDownMenu/LibUIDropDownMenu/LibUIDropDownMenuTemplates.xml
@@ -0,0 +1,385 @@
+<!-- $Id: LibUIDropDownMenuTemplates.xml 34 2018-07-25 13:26:33Z arith $ -->
+<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">
+	<Include file="LibUIDropDownMenuTemplates.lua"/>
+	<Frame name="L_UIDropDownCustomMenuEntryTemplate" enableMouse="true" hidden="true" mixin="L_UIDropDownCustomMenuEntryMixin" virtual="true">
+		<Scripts>
+			<OnEnter method="OnEnter"/>
+			<OnLeave method="OnLeave"/>
+		</Scripts>
+	</Frame>
+
+	<Button name="L_UIDropDownMenuButtonTemplate" virtual="true">
+		<Size x="100" y="16"/>
+		<Layers>
+			<Layer level="BACKGROUND">
+				<Texture name="$parentHighlight" parentKey="Highlight" file="Interface\QuestFrame\UI-QuestTitleHighlight" alphaMode="ADD" setAllPoints="true" hidden="true"/>
+			</Layer>
+			<Layer level="ARTWORK">
+				<Texture name="$parentCheck" file="Interface\Common\UI-DropDownRadioChecks">
+					<Size x="16" y="16"/>
+					<Anchors>
+						<Anchor point="LEFT">
+							<Offset x="0" y="0"/>
+						</Anchor>
+					</Anchors>
+					<TexCoords left="0" right="0.5" top="0.5" bottom="1.0"/>
+				</Texture>
+				<Texture name="$parentUnCheck" file="Interface\Common\UI-DropDownRadioChecks">
+					<Size x="16" y="16"/>
+					<Anchors>
+						<Anchor point="LEFT">
+							<Offset x="0" y="0"/>
+						</Anchor>
+					</Anchors>
+					<TexCoords left="0.5" right="1.0" top="0.5" bottom="1.0"/>
+				</Texture>
+				<Texture parentKey="Icon" name="$parentIcon" hidden="true">
+					<Size>
+						<AbsDimension x="16" y="16"/>
+					</Size>
+					<Anchors>
+						<Anchor point="RIGHT">
+							<Offset x="0" y="0"/>
+						</Anchor>
+					</Anchors>
+				</Texture>
+			</Layer>
+		</Layers>
+		<Frames>
+			<Button name="$parentColorSwatch" hidden="true">
+				<Size>
+					<AbsDimension x="16" y="16"/>
+				</Size>
+				<Anchors>
+					<Anchor point="RIGHT">
+						<Offset>
+							<AbsDimension x="-6" y="0"/>
+						</Offset>
+					</Anchor>
+				</Anchors>
+				<Layers>
+					<Layer level="BACKGROUND">
+						<Texture name="$parentSwatchBg">
+							<Size>
+								<AbsDimension x="14" y="14"/>
+							</Size>
+							<Anchors>
+								<Anchor point="CENTER">
+									<Offset>
+										<AbsDimension x="0" y="0"/>
+									</Offset>
+								</Anchor>
+							</Anchors>
+							<Color r="1.0" g="1.0" b="1.0"/>
+						</Texture>
+					</Layer>
+				</Layers>
+				<Scripts>
+					<OnClick>
+						CloseMenus();
+						L_UIDropDownMenuButton_OpenColorPicker(self:GetParent());
+					</OnClick>
+					<OnEnter>
+						L_CloseDropDownMenus(self:GetParent():GetParent():GetID() + 1);
+						_G[self:GetName().."SwatchBg"]:SetVertexColor(NORMAL_FONT_COLOR.r, NORMAL_FONT_COLOR.g, NORMAL_FONT_COLOR.b);
+						L_UIDropDownMenu_StopCounting(self:GetParent():GetParent());
+					</OnEnter>
+					<OnLeave>
+						_G[self:GetName().."SwatchBg"]:SetVertexColor(HIGHLIGHT_FONT_COLOR.r, HIGHLIGHT_FONT_COLOR.g, HIGHLIGHT_FONT_COLOR.b);
+						L_UIDropDownMenu_StartCounting(self:GetParent():GetParent());
+					</OnLeave>
+				</Scripts>
+				<NormalTexture name="$parentNormalTexture" file="Interface\ChatFrame\ChatFrameColorSwatch"/>
+			</Button>
+			<Button name="$parentExpandArrow" hidden="true" motionScriptsWhileDisabled="true">
+				<Size>
+					<AbsDimension x="16" y="16"/>
+				</Size>
+				<Anchors>
+					<Anchor point="RIGHT">
+						<Offset>
+							<AbsDimension x="0" y="0"/>
+						</Offset>
+					</Anchor>
+				</Anchors>
+				<Scripts>
+					<OnClick>
+						L_ToggleDropDownMenu(self:GetParent():GetParent():GetID() + 1, self:GetParent().value, nil, nil, nil, nil, self:GetParent().menuList, self);
+					</OnClick>
+					<OnEnter>
+						local level =  self:GetParent():GetParent():GetID() + 1;
+
+						L_CloseDropDownMenus(level);
+
+						if self:IsEnabled() then
+							local listFrame = _G["L_DropDownList"..level];
+							if ( not listFrame or not listFrame:IsShown() or select(2, listFrame:GetPoint()) ~= self ) then
+								L_ToggleDropDownMenu(level, self:GetParent().value, nil, nil, nil, nil, self:GetParent().menuList, self);
+							end
+						end
+
+						L_UIDropDownMenu_StopCounting(self:GetParent():GetParent());
+					</OnEnter>
+					<OnLeave>
+						L_UIDropDownMenu_StartCounting(self:GetParent():GetParent());
+					</OnLeave>
+				</Scripts>
+				<NormalTexture file="Interface\ChatFrame\ChatFrameExpandArrow"/>
+			</Button>
+			<Button name="$parentInvisibleButton" hidden="true" parentKey="invisibleButton">
+				<Anchors>
+					<Anchor point="TOPLEFT"/>
+					<Anchor point="BOTTOMLEFT"/>
+					<Anchor point="RIGHT" relativeTo="$parentColorSwatch" relativePoint="LEFT">
+						<Offset>
+							<AbsDimension x="0" y="0"/>
+						</Offset>
+					</Anchor>
+				</Anchors>
+				<Scripts>
+					<OnEnter function="L_UIDropDownMenuButtonInvisibleButton_OnEnter"/>
+					<OnLeave function="L_UIDropDownMenuButtonInvisibleButton_OnLeave"/>
+				</Scripts>
+			</Button>
+		</Frames>
+		<Scripts>
+			<OnLoad>
+				self:SetFrameLevel(self:GetParent():GetFrameLevel()+2);
+			</OnLoad>
+			<OnClick>
+				L_UIDropDownMenuButton_OnClick(self, button, down);
+			</OnClick>
+			<OnEnter function="L_UIDropDownMenuButton_OnEnter"/>
+			<OnLeave function="L_UIDropDownMenuButton_OnLeave"/>
+			<OnEnable>
+				self.invisibleButton:Hide();
+			</OnEnable>
+			<OnDisable>
+				self.invisibleButton:Show();
+			</OnDisable>
+		</Scripts>
+		<ButtonText name="$parentNormalText">
+			<Anchors>
+				<Anchor point="LEFT">
+					<Offset x="-5" y="0"/>
+				</Anchor>
+			</Anchors>
+		</ButtonText>
+		<NormalFont style="GameFontHighlightSmallLeft"/>
+		<HighlightFont style="GameFontHighlightSmallLeft"/>
+		<DisabledFont style="GameFontDisableSmallLeft"/>
+	</Button>
+
+	<Button name="L_UIDropDownListTemplate" hidden="true" frameStrata="DIALOG" enableMouse="true" virtual="true">
+		<Frames>
+			<Frame name="$parentBackdrop" setAllPoints="true">
+				<Backdrop bgFile="Interface\DialogFrame\UI-DialogBox-Background-Dark" edgeFile="Interface\DialogFrame\UI-DialogBox-Border" tile="true">
+					<BackgroundInsets>
+						<AbsInset left="11" right="11" top="11" bottom="9"/>
+					</BackgroundInsets>
+					<TileSize>
+						<AbsValue val="32"/>
+					</TileSize>
+					<EdgeSize>
+						<AbsValue val="32"/>
+					</EdgeSize>
+				</Backdrop>
+			</Frame>
+			<Frame name="$parentMenuBackdrop" setAllPoints="true">
+				<Backdrop bgFile="Interface\Tooltips\UI-Tooltip-Background" edgeFile="Interface\Tooltips\UI-Tooltip-Border" tile="true">
+					<EdgeSize>
+						<AbsValue val="16"/>
+					</EdgeSize>
+					<TileSize>
+						<AbsValue val="16"/>
+					</TileSize>
+					<BackgroundInsets>
+						<AbsInset left="5" right="4" top="4" bottom="4"/>
+					</BackgroundInsets>
+				</Backdrop>
+				<Scripts>
+					<OnLoad>
+						self:SetBackdropBorderColor(TOOLTIP_DEFAULT_COLOR.r, TOOLTIP_DEFAULT_COLOR.g, TOOLTIP_DEFAULT_COLOR.b);
+						self:SetBackdropColor(TOOLTIP_DEFAULT_BACKGROUND_COLOR.r, TOOLTIP_DEFAULT_BACKGROUND_COLOR.g, TOOLTIP_DEFAULT_BACKGROUND_COLOR.b);
+					</OnLoad>
+				</Scripts>
+			</Frame>
+			<Button name="$parentButton1" inherits="L_UIDropDownMenuButtonTemplate" id="1"/>
+			<Button name="$parentButton2" inherits="L_UIDropDownMenuButtonTemplate" id="2"/>
+			<Button name="$parentButton3" inherits="L_UIDropDownMenuButtonTemplate" id="3"/>
+			<Button name="$parentButton4" inherits="L_UIDropDownMenuButtonTemplate" id="4"/>
+			<Button name="$parentButton5" inherits="L_UIDropDownMenuButtonTemplate" id="5"/>
+			<Button name="$parentButton6" inherits="L_UIDropDownMenuButtonTemplate" id="6"/>
+			<Button name="$parentButton7" inherits="L_UIDropDownMenuButtonTemplate" id="7"/>
+			<Button name="$parentButton8" inherits="L_UIDropDownMenuButtonTemplate" id="8"/>
+		</Frames>
+		<Scripts>
+			<OnClick>
+				self:Hide();
+			</OnClick>
+			<OnEnter>
+				L_UIDropDownMenu_StopCounting(self, motion);
+			</OnEnter>
+			<OnLeave>
+				L_UIDropDownMenu_StartCounting(self, motion);
+			</OnLeave>
+			<OnUpdate>
+				L_UIDropDownMenu_OnUpdate(self, elapsed);
+			</OnUpdate>
+			<OnShow>
+				for i=1, L_UIDROPDOWNMENU_MAXBUTTONS do
+					if (not self.noResize) then
+						_G[self:GetName().."Button"..i]:SetWidth(self.maxWidth);
+					end
+				end
+				if (not self.noResize) then
+					self:SetWidth(self.maxWidth+25);
+				end
+				self.showTimer = nil;
+				if ( self:GetID() > 1 ) then
+					self.parent = _G["L_DropDownList"..(self:GetID() - 1)];
+				end
+			</OnShow>
+			<OnHide>
+				L_UIDropDownMenu_OnHide(self);
+			</OnHide>
+		</Scripts>
+	</Button>
+
+	<Frame name="L_UIDropDownMenuTemplate" virtual="true">
+		<Size>
+			<AbsDimension x="40" y="32"/>
+		</Size>
+		<Layers>
+			<Layer level="ARTWORK">
+				<Texture name="$parentLeft" parentKey="Left" file="Interface\Glues\CharacterCreate\CharacterCreate-LabelFrame">
+					<Size>
+						<AbsDimension x="25" y="64"/>
+					</Size>
+					<Anchors>
+						<Anchor point="TOPLEFT">
+							<Offset>
+								<AbsDimension x="0" y="17"/>
+							</Offset>
+						</Anchor>
+					</Anchors>
+					<TexCoords left="0" right="0.1953125" top="0" bottom="1"/>
+				</Texture>
+				<Texture name="$parentMiddle" parentKey="Middle" file="Interface\Glues\CharacterCreate\CharacterCreate-LabelFrame">
+					<Size>
+						<AbsDimension x="115" y="64"/>
+					</Size>
+					<Anchors>
+						<Anchor point="LEFT" relativeKey="$parent.Left" relativePoint="RIGHT"/>
+					</Anchors>
+					<TexCoords left="0.1953125" right="0.8046875" top="0" bottom="1"/>
+				</Texture>
+				<Texture name="$parentRight" parentKey="Right" file="Interface\Glues\CharacterCreate\CharacterCreate-LabelFrame">
+					<Size>
+						<AbsDimension x="25" y="64"/>
+					</Size>
+					<Anchors>
+						<Anchor point="LEFT" relativeKey="$parent.Middle" relativePoint="RIGHT"/>
+					</Anchors>
+					<TexCoords left="0.8046875" right="1" top="0" bottom="1"/>
+				</Texture>
+				<FontString parentKey="Text" name="$parentText" inherits="GameFontHighlightSmall" wordwrap="false" justifyH="RIGHT">
+					<Size>
+						<AbsDimension x="0" y="10"/>
+					</Size>
+					<Anchors>
+						<Anchor point="RIGHT" relativeKey="$parent.Right">
+							<Offset>
+								<AbsDimension x="-43" y="2"/>
+							</Offset>
+						</Anchor>
+					</Anchors>
+				</FontString>
+			</Layer>
+			<Layer level="OVERLAY">
+				<Texture parentKey="Icon" name="$parentIcon" hidden="true">
+					<Size>
+						<AbsDimension x="16" y="16"/>
+					</Size>
+					<Anchors>
+						<Anchor point="LEFT">
+							<Offset x="30" y="2"/>
+						</Anchor>
+					</Anchors>
+				</Texture>
+			</Layer>
+		</Layers>
+		<Frames>
+			<Button parentKey="Button" name="$parentButton" motionScriptsWhileDisabled="true" >
+				<Size>
+					<AbsDimension x="24" y="24"/>
+				</Size>
+				<Anchors>
+					<Anchor point="TOPRIGHT" relativeKey="$parent.Right">
+						<Offset>
+							<AbsDimension x="-16" y="-18"/>
+						</Offset>
+					</Anchor>
+				</Anchors>
+				<Scripts>
+					<OnEnter>
+						local parent = self:GetParent();
+						local myscript = parent:GetScript("OnEnter");
+						if(myscript ~= nil) then
+							myscript(parent);
+						end
+					</OnEnter>
+					<OnLeave>
+						local parent = self:GetParent();
+						local myscript = parent:GetScript("OnLeave");
+						if(myscript ~= nil) then
+							myscript(parent);
+						end
+					</OnLeave>
+					<OnClick>
+						L_ToggleDropDownMenu(nil, nil, self:GetParent());
+						PlaySound(SOUNDKIT.IG_MAINMENU_OPTION_CHECKBOX_ON);
+					</OnClick>
+				</Scripts>
+				<NormalTexture name="$parentNormalTexture" parentKey="NormalTexture" file="Interface\ChatFrame\UI-ChatIcon-ScrollDown-Up">
+					<Size>
+						<AbsDimension x="24" y="24"/>
+					</Size>
+					<Anchors>
+						<Anchor point="RIGHT"/>
+					</Anchors>
+				</NormalTexture>
+				<PushedTexture name="$parentPushedTexture" parentKey="PushedTexture" file="Interface\ChatFrame\UI-ChatIcon-ScrollDown-Down">
+					<Size>
+						<AbsDimension x="24" y="24"/>
+					</Size>
+					<Anchors>
+						<Anchor point="RIGHT"/>
+					</Anchors>
+				</PushedTexture>
+				<DisabledTexture name="$parentDisabledTexture" parentKey="DisabledTexture" file="Interface\ChatFrame\UI-ChatIcon-ScrollDown-Disabled">
+					<Size>
+						<AbsDimension x="24" y="24"/>
+					</Size>
+					<Anchors>
+						<Anchor point="RIGHT"/>
+					</Anchors>
+				</DisabledTexture>
+				<HighlightTexture name="$parentHighlightTexture" parentKey="HighlightTexture" file="Interface\Buttons\UI-Common-MouseHilight" alphaMode="ADD">
+					<Size>
+						<AbsDimension x="24" y="24"/>
+					</Size>
+					<Anchors>
+						<Anchor point="RIGHT"/>
+					</Anchors>
+				</HighlightTexture>
+			</Button>
+		</Frames>
+		<Scripts>
+			<OnHide>
+				L_CloseDropDownMenus();
+			</OnHide>
+		</Scripts>
+	</Frame>
+</Ui>
diff --git a/Titan/libs/Ace3.lua b/Titan/libs/Ace3.lua
new file mode 100755
index 0000000..fb0a3fa
--- /dev/null
+++ b/Titan/libs/Ace3.lua
@@ -0,0 +1,110 @@
+
+-- This file is only there in standalone Ace3 and provides handy dev tool stuff I guess
+-- for now only /rl to reload your UI :)
+-- note the complete overkill use of AceAddon and console, ain't it cool?
+
+-- GLOBALS: next, loadstring, ReloadUI, geterrorhandler
+-- GLOBALS: BINDING_HEADER_ACE3, BINDING_NAME_RELOADUI, Ace3, LibStub
+
+-- BINDINGs labels
+BINDING_HEADER_ACE3 = "Ace3"
+BINDING_NAME_RELOADUI = "ReloadUI"
+--
+
+local gui = LibStub("AceGUI-3.0")
+local reg = LibStub("AceConfigRegistry-3.0")
+local dialog = LibStub("AceConfigDialog-3.0")
+
+Ace3 = LibStub("AceAddon-3.0"):NewAddon("Ace3", "AceConsole-3.0")
+local Ace3 = Ace3
+
+local selectedgroup
+local frame
+local select
+local status = {}
+local configs = {}
+
+local function frameOnClose()
+	gui:Release(frame)
+	frame = nil
+end
+
+local function RefreshConfigs()
+	for name in reg:IterateOptionsTables() do
+		configs[name] = name
+	end
+end
+
+local function ConfigSelected(widget, event, value)
+	selectedgroup = value
+	dialog:Open(value, widget)
+end
+
+local old_CloseSpecialWindows
+
+-- GLOBALS: CloseSpecialWindows, next
+function Ace3:Open()
+	if not old_CloseSpecialWindows then
+		old_CloseSpecialWindows = CloseSpecialWindows
+		CloseSpecialWindows = function()
+			local found = old_CloseSpecialWindows()
+			if frame then
+				frame:Hide()
+				return true
+			end
+			return found
+		end
+	end
+	RefreshConfigs()
+	if next(configs) == nil then
+		self:Print("No Configs are Registered")
+		return
+	end
+
+	if not frame then
+		frame = gui:Create("Frame")
+		frame:ReleaseChildren()
+		frame:SetTitle("Ace3 Options")
+		frame:SetLayout("FILL")
+		frame:SetCallback("OnClose", frameOnClose)
+
+		select = gui:Create("DropdownGroup")
+		select:SetGroupList(configs)
+		select:SetCallback("OnGroupSelected", ConfigSelected)
+		frame:AddChild(select)
+	end
+	if not selectedgroup then
+		selectedgroup = next(configs)
+	end
+	select:SetGroup(selectedgroup)
+	frame:Show()
+end
+
+local function RefreshOnUpdate(this)
+	select:SetGroup(selectedgroup)
+	this:SetScript("OnUpdate", nil)
+end
+
+function Ace3:ConfigTableChanged(event, appName)
+	if selectedgroup == appName and frame then
+		frame.frame:SetScript("OnUpdate", RefreshOnUpdate)
+	end
+end
+
+reg.RegisterCallback(Ace3, "ConfigTableChange", "ConfigTableChanged")
+
+function Ace3:PrintCmd(input)
+	input = input:trim():match("^(.-);*$")
+	local func, err = loadstring("LibStub(\"AceConsole-3.0\"):Print(" .. input .. ")")
+	if not func then
+		LibStub("AceConsole-3.0"):Print("Error: " .. err)
+	else
+		func()
+	end
+end
+
+function Ace3:OnInitialize()
+	self:RegisterChatCommand("ace3", function() self:Open() end)
+	self:RegisterChatCommand("rl", function() ReloadUI() end)
+	self:RegisterChatCommand("print", "PrintCmd")
+end
diff --git a/Titan/libs/Ace3.toc b/Titan/libs/Ace3.toc
new file mode 100755
index 0000000..5099d33
--- /dev/null
+++ b/Titan/libs/Ace3.toc
@@ -0,0 +1,27 @@
+## Interface: 80000
+
+## Title: Lib: Ace3
+## Notes: AddOn development framework
+## Author: Ace3 Development Team
+## X-Website: http://www.wowace.com
+## X-Category: Library
+## X-License: Limited BSD
+
+LibStub\LibStub.lua
+CallbackHandler-1.0\CallbackHandler-1.0.xml
+AceAddon-3.0\AceAddon-3.0.xml
+AceEvent-3.0\AceEvent-3.0.xml
+AceTimer-3.0\AceTimer-3.0.xml
+AceBucket-3.0\AceBucket-3.0.xml
+AceHook-3.0\AceHook-3.0.xml
+AceDB-3.0\AceDB-3.0.xml
+AceDBOptions-3.0\AceDBOptions-3.0.xml
+AceLocale-3.0\AceLocale-3.0.xml
+AceConsole-3.0\AceConsole-3.0.xml
+AceGUI-3.0\AceGUI-3.0.xml
+AceConfig-3.0\AceConfig-3.0.xml
+AceComm-3.0\AceComm-3.0.xml
+AceTab-3.0\AceTab-3.0.xml
+AceSerializer-3.0\AceSerializer-3.0.xml
+
+Ace3.lua
diff --git a/Titan/libs/AceAddon-3.0/AceAddon-3.0.lua b/Titan/libs/AceAddon-3.0/AceAddon-3.0.lua
new file mode 100755
index 0000000..b338695
--- /dev/null
+++ b/Titan/libs/AceAddon-3.0/AceAddon-3.0.lua
@@ -0,0 +1,674 @@
+--- **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 1084 2013-04-27 20:14:11Z 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 CreateDispatcher(argCount)
+	local code = [[
+		local xpcall, eh = ...
+		local method, ARGS
+		local function call() return method(ARGS) end
+
+		local function dispatch(func, ...)
+			 method = func
+			 if not method then return end
+			 ARGS = ...
+			 return xpcall(call, eh)
+		end
+
+		return dispatch
+	]]
+
+	local ARGS = {}
+	for i = 1, argCount do ARGS[i] = "arg"..i end
+	code = code:gsub("ARGS", tconcat(ARGS, ", "))
+	return assert(loadstring(code, "safecall Dispatcher["..argCount.."]"))(xpcall, errorhandler)
+end
+
+local Dispatchers = setmetatable({}, {__index=function(self, argCount)
+	local dispatcher = CreateDispatcher(argCount)
+	rawset(self, argCount, dispatcher)
+	return dispatcher
+end})
+Dispatchers[0] = function(func)
+	return xpcall(func, errorhandler)
+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 Dispatchers[select('#', ...)](func, ...)
+	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
diff --git a/Titan/libs/AceAddon-3.0/AceAddon-3.0.xml b/Titan/libs/AceAddon-3.0/AceAddon-3.0.xml
new file mode 100755
index 0000000..c607008
--- /dev/null
+++ b/Titan/libs/AceAddon-3.0/AceAddon-3.0.xml
@@ -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>
diff --git a/Titan/libs/AceConfig-3.0/AceConfig-3.0.lua b/Titan/libs/AceConfig-3.0/AceConfig-3.0.lua
new file mode 100755
index 0000000..c2ac7cc
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfig-3.0.lua
@@ -0,0 +1,58 @@
+--- AceConfig-3.0 wrapper library.
+-- Provides an API to register an options table with the config registry,
+-- as well as associate it with a slash command.
+-- @class file
+-- @name AceConfig-3.0
+-- @release $Id: AceConfig-3.0.lua 1161 2017-08-12 14:30:16Z funkydude $
+
+--[[
+AceConfig-3.0
+
+Very light wrapper library that combines all the AceConfig subcomponents into one more easily used whole.
+
+]]
+
+local cfgreg = LibStub("AceConfigRegistry-3.0")
+local cfgcmd = LibStub("AceConfigCmd-3.0")
+
+local MAJOR, MINOR = "AceConfig-3.0", 3
+local AceConfig = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not AceConfig then return end
+
+--TODO: local cfgdlg = LibStub("AceConfigDialog-3.0", true)
+--TODO: local cfgdrp = LibStub("AceConfigDropdown-3.0", true)
+
+-- Lua APIs
+local pcall, error, type, pairs = pcall, error, type, pairs
+
+-- -------------------------------------------------------------------
+-- :RegisterOptionsTable(appName, options, slashcmd, persist)
+--
+-- - appName - (string) application name
+-- - options - table or function ref, see AceConfigRegistry
+-- - slashcmd - slash command (string) or table with commands, or nil to NOT create a slash command
+
+--- Register a option table with the AceConfig registry.
+-- You can supply a slash command (or a table of slash commands) to register with AceConfigCmd directly.
+-- @paramsig appName, options [, slashcmd]
+-- @param appName The application name for the config table.
+-- @param options The option table (or a function to generate one on demand).  http://www.wowace.com/addons/ace3/pages/ace-config-3-0-options-tables/
+-- @param slashcmd A slash command to register for the option table, or a table of slash commands.
+-- @usage
+-- local AceConfig = LibStub("AceConfig-3.0")
+-- AceConfig:RegisterOptionsTable("MyAddon", myOptions, {"/myslash", "/my"})
+function AceConfig:RegisterOptionsTable(appName, options, slashcmd)
+	local ok,msg = pcall(cfgreg.RegisterOptionsTable, self, appName, options)
+	if not ok then error(msg, 2) end
+
+	if slashcmd then
+		if type(slashcmd) == "table" then
+			for _,cmd in pairs(slashcmd) do
+				cfgcmd:CreateChatCommand(cmd, appName)
+			end
+		else
+			cfgcmd:CreateChatCommand(slashcmd, appName)
+		end
+	end
+end
diff --git a/Titan/libs/AceConfig-3.0/AceConfig-3.0.xml b/Titan/libs/AceConfig-3.0/AceConfig-3.0.xml
new file mode 100755
index 0000000..84c8d03
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfig-3.0.xml
@@ -0,0 +1,8 @@
+<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">
+	<Include file="AceConfigRegistry-3.0\AceConfigRegistry-3.0.xml"/>
+	<Include file="AceConfigCmd-3.0\AceConfigCmd-3.0.xml"/>
+	<Include file="AceConfigDialog-3.0\AceConfigDialog-3.0.xml"/>
+	<!--<Include file="AceConfigDropdown-3.0\AceConfigDropdown-3.0.xml"/>-->
+	<Script file="AceConfig-3.0.lua"/>
+</Ui>
diff --git a/Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.lua b/Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.lua
new file mode 100755
index 0000000..7902fd1
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.lua
@@ -0,0 +1,794 @@
+--- AceConfigCmd-3.0 handles access to an options table through the "command line" interface via the ChatFrames.
+-- @class file
+-- @name AceConfigCmd-3.0
+-- @release $Id: AceConfigCmd-3.0.lua 1161 2017-08-12 14:30:16Z funkydude $
+
+--[[
+AceConfigCmd-3.0
+
+Handles commandline optionstable access
+
+REQUIRES: AceConsole-3.0 for command registration (loaded on demand)
+
+]]
+
+-- TODO: plugin args
+
+local cfgreg = LibStub("AceConfigRegistry-3.0")
+
+local MAJOR, MINOR = "AceConfigCmd-3.0", 14
+local AceConfigCmd = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not AceConfigCmd then return end
+
+AceConfigCmd.commands = AceConfigCmd.commands or {}
+local commands = AceConfigCmd.commands
+
+local AceConsole -- LoD
+local AceConsoleName = "AceConsole-3.0"
+
+-- Lua APIs
+local strsub, strsplit, strlower, strmatch, strtrim = string.sub, string.split, string.lower, string.match, string.trim
+local format, tonumber, tostring = string.format, tonumber, tostring
+local tsort, tinsert = table.sort, table.insert
+local select, pairs, next, type = select, pairs, next, type
+local error, assert = error, assert
+
+-- WoW APIs
+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: LibStub, SELECTED_CHAT_FRAME, DEFAULT_CHAT_FRAME
+
+
+local L = setmetatable({}, {	-- TODO: replace with proper locale
+	__index = function(self,k) return k end
+})
+
+
+
+local function print(msg)
+	(SELECTED_CHAT_FRAME or DEFAULT_CHAT_FRAME):AddMessage(msg)
+end
+
+-- constants used by getparam() calls below
+
+local handlertypes = {["table"]=true}
+local handlermsg = "expected a table"
+
+local functypes = {["function"]=true, ["string"]=true}
+local funcmsg = "expected function or member name"
+
+
+-- pickfirstset() - picks the first non-nil value and returns it
+
+local function pickfirstset(...)
+	for i=1,select("#",...) do
+		if select(i,...)~=nil then
+			return select(i,...)
+		end
+	end
+end
+
+
+-- err() - produce real error() regarding malformed options tables etc
+
+local function err(info,inputpos,msg )
+	local cmdstr=" "..strsub(info.input, 1, inputpos-1)
+	error(MAJOR..": /" ..info[0] ..cmdstr ..": "..(msg or "malformed options table"), 2)
+end
+
+
+-- usererr() - produce chatframe message regarding bad slash syntax etc
+
+local function usererr(info,inputpos,msg )
+	local cmdstr=strsub(info.input, 1, inputpos-1);
+	print("/" ..info[0] .. " "..cmdstr ..": "..(msg or "malformed options table"))
+end
+
+
+-- callmethod() - call a given named method (e.g. "get", "set") with given arguments
+
+local function callmethod(info, inputpos, tab, methodtype, ...)
+	local method = info[methodtype]
+	if not method then
+		err(info, inputpos, "'"..methodtype.."': not set")
+	end
+
+	info.arg = tab.arg
+	info.option = tab
+	info.type = tab.type
+
+	if type(method)=="function" then
+		return method(info, ...)
+	elseif type(method)=="string" then
+		if type(info.handler[method])~="function" then
+			err(info, inputpos, "'"..methodtype.."': '"..method.."' is not a member function of "..tostring(info.handler))
+		end
+		return info.handler[method](info.handler, info, ...)
+	else
+		assert(false)	-- type should have already been checked on read
+	end
+end
+
+-- callfunction() - call a given named function (e.g. "name", "desc") with given arguments
+
+local function callfunction(info, tab, methodtype, ...)
+	local method = tab[methodtype]
+
+	info.arg = tab.arg
+	info.option = tab
+	info.type = tab.type
+
+	if type(method)=="function" then
+		return method(info, ...)
+	else
+		assert(false) -- type should have already been checked on read
+	end
+end
+
+-- do_final() - do the final step (set/execute) along with validation and confirmation
+
+local function do_final(info, inputpos, tab, methodtype, ...)
+	if info.validate then
+		local res = callmethod(info,inputpos,tab,"validate",...)
+		if type(res)=="string" then
+			usererr(info, inputpos, "'"..strsub(info.input, inputpos).."' - "..res)
+			return
+		end
+	end
+	-- console ignores .confirm
+
+	callmethod(info,inputpos,tab,methodtype, ...)
+end
+
+
+-- getparam() - used by handle() to retreive and store "handler", "get", "set", etc
+
+local function getparam(info, inputpos, tab, depth, paramname, types, errormsg)
+	local old,oldat = info[paramname], info[paramname.."_at"]
+	local val=tab[paramname]
+	if val~=nil then
+		if val==false then
+			val=nil
+		elseif not types[type(val)] then
+			err(info, inputpos, "'" .. paramname.. "' - "..errormsg)
+		end
+		info[paramname] = val
+		info[paramname.."_at"] = depth
+	end
+	return old,oldat
+end
+
+
+-- iterateargs(tab) - custom iterator that iterates both t.args and t.plugins.*
+local dummytable={}
+
+local function iterateargs(tab)
+	if not tab.plugins then
+		return pairs(tab.args)
+	end
+
+	local argtabkey,argtab=next(tab.plugins)
+	local v
+
+	return function(_, k)
+		while argtab do
+			k,v = next(argtab, k)
+			if k then return k,v end
+			if argtab==tab.args then
+				argtab=nil
+			else
+				argtabkey,argtab = next(tab.plugins, argtabkey)
+				if not argtabkey then
+					argtab=tab.args
+				end
+			end
+		end
+	end
+end
+
+local function checkhidden(info, inputpos, tab)
+	if tab.cmdHidden~=nil then
+		return tab.cmdHidden
+	end
+	local hidden = tab.hidden
+	if type(hidden) == "function" or type(hidden) == "string" then
+		info.hidden = hidden
+		hidden = callmethod(info, inputpos, tab, 'hidden')
+		info.hidden = nil
+	end
+	return hidden
+end
+
+local function showhelp(info, inputpos, tab, depth, noHead)
+	if not noHead then
+		print("|cff33ff99"..info.appName.."|r: Arguments to |cffffff78/"..info[0].."|r "..strsub(info.input,1,inputpos-1)..":")
+	end
+
+	local sortTbl = {}	-- [1..n]=name
+	local refTbl = {}   -- [name]=tableref
+
+	for k,v in iterateargs(tab) do
+		if not refTbl[k] then	-- a plugin overriding something in .args
+			tinsert(sortTbl, k)
+			refTbl[k] = v
+		end
+	end
+
+	tsort(sortTbl, function(one, two)
+		local o1 = refTbl[one].order or 100
+		local o2 = refTbl[two].order or 100
+		if type(o1) == "function" or type(o1) == "string" then
+			info.order = o1
+			info[#info+1] = one
+			o1 = callmethod(info, inputpos, refTbl[one], "order")
+			info[#info] = nil
+			info.order = nil
+		end
+		if type(o2) == "function" or type(o1) == "string" then
+			info.order = o2
+			info[#info+1] = two
+			o2 = callmethod(info, inputpos, refTbl[two], "order")
+			info[#info] = nil
+			info.order = nil
+		end
+		if o1<0 and o2<0 then return o1<o2 end
+		if o2<0 then return true end
+		if o1<0 then return false end
+		if o1==o2 then return tostring(one)<tostring(two) end   -- compare names
+		return o1<o2
+	end)
+
+	for i = 1, #sortTbl do
+		local k = sortTbl[i]
+		local v = refTbl[k]
+		if not checkhidden(info, inputpos, v) then
+			if v.type ~= "description" and v.type ~= "header" then
+				-- recursively show all inline groups
+				local name, desc = v.name, v.desc
+				if type(name) == "function" then
+					name = callfunction(info, v, 'name')
+				end
+				if type(desc) == "function" then
+					desc = callfunction(info, v, 'desc')
+				end
+				if v.type == "group" and pickfirstset(v.cmdInline, v.inline, false) then
+					print("  "..(desc or name)..":")
+					local oldhandler,oldhandler_at = getparam(info, inputpos, v, depth, "handler", handlertypes, handlermsg)
+					showhelp(info, inputpos, v, depth, true)
+					info.handler,info.handler_at = oldhandler,oldhandler_at
+				else
+					local key = k:gsub(" ", "_")
+					print("  |cffffff78"..key.."|r - "..(desc or name or ""))
+				end
+			end
+		end
+	end
+end
+
+
+local function keybindingValidateFunc(text)
+	if text == nil or text == "NONE" then
+		return nil
+	end
+	text = text:upper()
+	local shift, ctrl, alt
+	local modifier
+	while true do
+		if text == "-" then
+			break
+		end
+		modifier, text = strsplit('-', text, 2)
+		if text then
+			if modifier ~= "SHIFT" and modifier ~= "CTRL" and modifier ~= "ALT" then
+				return false
+			end
+			if modifier == "SHIFT" then
+				if shift then
+					return false
+				end
+				shift = true
+			end
+			if modifier == "CTRL" then
+				if ctrl then
+					return false
+				end
+				ctrl = true
+			end
+			if modifier == "ALT" then
+				if alt then
+					return false
+				end
+				alt = true
+			end
+		else
+			text = modifier
+			break
+		end
+	end
+	if text == "" then
+		return false
+	end
+	if not text:find("^F%d+$") and text ~= "CAPSLOCK" and text:len() ~= 1 and (text:byte() < 128 or text:len() > 4) and not _G["KEY_" .. text] then
+		return false
+	end
+	local s = text
+	if shift then
+		s = "SHIFT-" .. s
+	end
+	if ctrl then
+		s = "CTRL-" .. s
+	end
+	if alt then
+		s = "ALT-" .. s
+	end
+	return s
+end
+
+-- handle() - selfrecursing function that processes input->optiontable
+-- - depth - starts at 0
+-- - retfalse - return false rather than produce error if a match is not found (used by inlined groups)
+
+local function handle(info, inputpos, tab, depth, retfalse)
+
+	if not(type(tab)=="table" and type(tab.type)=="string") then err(info,inputpos) end
+
+	-------------------------------------------------------------------
+	-- Grab hold of handler,set,get,func,etc if set (and remember old ones)
+	-- Note that we do NOT validate if method names are correct at this stage,
+	-- the handler may change before they're actually used!
+
+	local oldhandler,oldhandler_at = getparam(info,inputpos,tab,depth,"handler",handlertypes,handlermsg)
+	local oldset,oldset_at = getparam(info,inputpos,tab,depth,"set",functypes,funcmsg)
+	local oldget,oldget_at = getparam(info,inputpos,tab,depth,"get",functypes,funcmsg)
+	local oldfunc,oldfunc_at = getparam(info,inputpos,tab,depth,"func",functypes,funcmsg)
+	local oldvalidate,oldvalidate_at = getparam(info,inputpos,tab,depth,"validate",functypes,funcmsg)
+	--local oldconfirm,oldconfirm_at = getparam(info,inputpos,tab,depth,"confirm",functypes,funcmsg)
+
+	-------------------------------------------------------------------
+	-- Act according to .type of this table
+
+	if tab.type=="group" then
+		------------ group --------------------------------------------
+
+		if type(tab.args)~="table" then err(info, inputpos) end
+		if tab.plugins and type(tab.plugins)~="table" then err(info,inputpos) end
+
+		-- grab next arg from input
+		local _,nextpos,arg = (info.input):find(" *([^ ]+) *", inputpos)
+		if not arg then
+			showhelp(info, inputpos, tab, depth)
+			return
+		end
+		nextpos=nextpos+1
+
+		-- loop .args and try to find a key with a matching name
+		for k,v in iterateargs(tab) do
+			if not(type(k)=="string" and type(v)=="table" and type(v.type)=="string") then err(info,inputpos, "options table child '"..tostring(k).."' is malformed") end
+
+			-- is this child an inline group? if so, traverse into it
+			if v.type=="group" and pickfirstset(v.cmdInline, v.inline, false) then
+				info[depth+1] = k
+				if handle(info, inputpos, v, depth+1, true)==false then
+					info[depth+1] = nil
+					-- wasn't found in there, but that's ok, we just keep looking down here
+				else
+					return	-- done, name was found in inline group
+				end
+			-- matching name and not a inline group
+			elseif strlower(arg)==strlower(k:gsub(" ", "_")) then
+				info[depth+1] = k
+				return handle(info,nextpos,v,depth+1)
+			end
+		end
+
+		-- no match
+		if retfalse then
+			-- restore old infotable members and return false to indicate failure
+			info.handler,info.handler_at = oldhandler,oldhandler_at
+			info.set,info.set_at = oldset,oldset_at
+			info.get,info.get_at = oldget,oldget_at
+			info.func,info.func_at = oldfunc,oldfunc_at
+			info.validate,info.validate_at = oldvalidate,oldvalidate_at
+			--info.confirm,info.confirm_at = oldconfirm,oldconfirm_at
+			return false
+		end
+
+		-- couldn't find the command, display error
+		usererr(info, inputpos, "'"..arg.."' - " .. L["unknown argument"])
+		return
+	end
+
+	local str = strsub(info.input,inputpos);
+
+	if tab.type=="execute" then
+		------------ execute --------------------------------------------
+		do_final(info, inputpos, tab, "func")
+
+
+
+	elseif tab.type=="input" then
+		------------ input --------------------------------------------
+
+		local res = true
+		if tab.pattern then
+			if not(type(tab.pattern)=="string") then err(info, inputpos, "'pattern' - expected a string") end
+			if not strmatch(str, tab.pattern) then
+				usererr(info, inputpos, "'"..str.."' - " .. L["invalid input"])
+				return
+			end
+		end
+
+		do_final(info, inputpos, tab, "set", str)
+
+
+
+	elseif tab.type=="toggle" then
+		------------ toggle --------------------------------------------
+		local b
+		local str = strtrim(strlower(str))
+		if str=="" then
+			b = callmethod(info, inputpos, tab, "get")
+
+			if tab.tristate then
+				--cycle in true, nil, false order
+				if b then
+					b = nil
+				elseif b == nil then
+					b = false
+				else
+					b = true
+				end
+			else
+				b = not b
+			end
+
+		elseif str==L["on"] then
+			b = true
+		elseif str==L["off"] then
+			b = false
+		elseif tab.tristate and str==L["default"] then
+			b = nil
+		else
+			if tab.tristate then
+				usererr(info, inputpos, format(L["'%s' - expected 'on', 'off' or 'default', or no argument to toggle."], str))
+			else
+				usererr(info, inputpos, format(L["'%s' - expected 'on' or 'off', or no argument to toggle."], str))
+			end
+			return
+		end
+
+		do_final(info, inputpos, tab, "set", b)
+
+
+	elseif tab.type=="range" then
+		------------ range --------------------------------------------
+		local val = tonumber(str)
+		if not val then
+			usererr(info, inputpos, "'"..str.."' - "..L["expected number"])
+			return
+		end
+		if type(info.step)=="number" then
+			val = val- (val % info.step)
+		end
+		if type(info.min)=="number" and val<info.min then
+			usererr(info, inputpos, val.." - "..format(L["must be equal to or higher than %s"], tostring(info.min)) )
+			return
+		end
+		if type(info.max)=="number" and val>info.max then
+			usererr(info, inputpos, val.." - "..format(L["must be equal to or lower than %s"], tostring(info.max)) )
+			return
+		end
+
+		do_final(info, inputpos, tab, "set", val)
+
+
+	elseif tab.type=="select" then
+		------------ select ------------------------------------
+		local str = strtrim(strlower(str))
+
+		local values = tab.values
+		if type(values) == "function" or type(values) == "string" then
+			info.values = values
+			values = callmethod(info, inputpos, tab, "values")
+			info.values = nil
+		end
+
+		if str == "" then
+			local b = callmethod(info, inputpos, tab, "get")
+			local fmt = "|cffffff78- [%s]|r %s"
+			local fmt_sel = "|cffffff78- [%s]|r %s |cffff0000*|r"
+			print(L["Options for |cffffff78"..info[#info].."|r:"])
+			for k, v in pairs(values) do
+				if b == k then
+					print(fmt_sel:format(k, v))
+				else
+					print(fmt:format(k, v))
+				end
+			end
+			return
+		end
+
+		local ok
+		for k,v in pairs(values) do
+			if strlower(k)==str then
+				str = k	-- overwrite with key (in case of case mismatches)
+				ok = true
+				break
+			end
+		end
+		if not ok then
+			usererr(info, inputpos, "'"..str.."' - "..L["unknown selection"])
+			return
+		end
+
+		do_final(info, inputpos, tab, "set", str)
+
+	elseif tab.type=="multiselect" then
+		------------ multiselect -------------------------------------------
+		local str = strtrim(strlower(str))
+
+		local values = tab.values
+		if type(values) == "function" or type(values) == "string" then
+			info.values = values
+			values = callmethod(info, inputpos, tab, "values")
+			info.values = nil
+		end
+
+		if str == "" then
+			local fmt = "|cffffff78- [%s]|r %s"
+			local fmt_sel = "|cffffff78- [%s]|r %s |cffff0000*|r"
+			print(L["Options for |cffffff78"..info[#info].."|r (multiple possible):"])
+			for k, v in pairs(values) do
+				if callmethod(info, inputpos, tab, "get", k) then
+					print(fmt_sel:format(k, v))
+				else
+					print(fmt:format(k, v))
+				end
+			end
+			return
+		end
+
+		--build a table of the selections, checking that they exist
+		--parse for =on =off =default in the process
+		--table will be key = true for options that should toggle, key = [on|off|default] for options to be set
+		local sels = {}
+		for v in str:gmatch("[^ ]+") do
+			--parse option=on etc
+			local opt, val = v:match('(.+)=(.+)')
+			--get option if toggling
+			if not opt then
+				opt = v
+			end
+
+			--check that the opt is valid
+			local ok
+			for k,v in pairs(values) do
+				if strlower(k)==opt then
+					opt = k	-- overwrite with key (in case of case mismatches)
+					ok = true
+					break
+				end
+			end
+
+			if not ok then
+				usererr(info, inputpos, "'"..opt.."' - "..L["unknown selection"])
+				return
+			end
+
+			--check that if val was supplied it is valid
+			if val then
+				if val == L["on"] or val == L["off"] or (tab.tristate and val == L["default"]) then
+					--val is valid insert it
+					sels[opt] = val
+				else
+					if tab.tristate then
+						usererr(info, inputpos, format(L["'%s' '%s' - expected 'on', 'off' or 'default', or no argument to toggle."], v, val))
+					else
+						usererr(info, inputpos, format(L["'%s' '%s' - expected 'on' or 'off', or no argument to toggle."], v, val))
+					end
+					return
+				end
+			else
+				-- no val supplied, toggle
+				sels[opt] = true
+			end
+		end
+
+		for opt, val in pairs(sels) do
+			local newval
+
+			if (val == true) then
+				--toggle the option
+				local b = callmethod(info, inputpos, tab, "get", opt)
+
+				if tab.tristate then
+					--cycle in true, nil, false order
+					if b then
+						b = nil
+					elseif b == nil then
+						b = false
+					else
+						b = true
+					end
+				else
+					b = not b
+				end
+				newval = b
+			else
+				--set the option as specified
+				if val==L["on"] then
+					newval = true
+				elseif val==L["off"] then
+					newval = false
+				elseif val==L["default"] then
+					newval = nil
+				end
+			end
+
+			do_final(info, inputpos, tab, "set", opt, newval)
+		end
+
+
+	elseif tab.type=="color" then
+		------------ color --------------------------------------------
+		local str = strtrim(strlower(str))
+		if str == "" then
+			--TODO: Show current value
+			return
+		end
+
+		local r, g, b, a
+
+		local hasAlpha = tab.hasAlpha
+		if type(hasAlpha) == "function" or type(hasAlpha) == "string" then
+			info.hasAlpha = hasAlpha
+			hasAlpha = callmethod(info, inputpos, tab, 'hasAlpha')
+			info.hasAlpha = nil
+		end
+
+		if hasAlpha then
+			if str:len() == 8 and str:find("^%x*$")  then
+				--parse a hex string
+				r,g,b,a = tonumber(str:sub(1, 2), 16) / 255, tonumber(str:sub(3, 4), 16) / 255, tonumber(str:sub(5, 6), 16) / 255, tonumber(str:sub(7, 8), 16) / 255
+			else
+				--parse seperate values
+				r,g,b,a = str:match("^([%d%.]+) ([%d%.]+) ([%d%.]+) ([%d%.]+)$")
+				r,g,b,a = tonumber(r), tonumber(g), tonumber(b), tonumber(a)
+			end
+			if not (r and g and b and a) then
+				usererr(info, inputpos, format(L["'%s' - expected 'RRGGBBAA' or 'r g b a'."], str))
+				return
+			end
+
+			if r >= 0.0 and r <= 1.0 and g >= 0.0 and g <= 1.0 and b >= 0.0 and b <= 1.0 and a >= 0.0 and a <= 1.0 then
+				--values are valid
+			elseif r >= 0 and r <= 255 and g >= 0 and g <= 255 and b >= 0 and b <= 255 and a >= 0 and a <= 255 then
+				--values are valid 0..255, convert to 0..1
+				r = r / 255
+				g = g / 255
+				b = b / 255
+				a = a / 255
+			else
+				--values are invalid
+				usererr(info, inputpos, format(L["'%s' - values must all be either in the range 0..1 or 0..255."], str))
+			end
+		else
+			a = 1.0
+			if str:len() == 6 and str:find("^%x*$") then
+				--parse a hex string
+				r,g,b = tonumber(str:sub(1, 2), 16) / 255, tonumber(str:sub(3, 4), 16) / 255, tonumber(str:sub(5, 6), 16) / 255
+			else
+				--parse seperate values
+				r,g,b = str:match("^([%d%.]+) ([%d%.]+) ([%d%.]+)$")
+				r,g,b = tonumber(r), tonumber(g), tonumber(b)
+			end
+			if not (r and g and b) then
+				usererr(info, inputpos, format(L["'%s' - expected 'RRGGBB' or 'r g b'."], str))
+				return
+			end
+			if r >= 0.0 and r <= 1.0 and g >= 0.0 and g <= 1.0 and b >= 0.0 and b <= 1.0 then
+				--values are valid
+			elseif r >= 0 and r <= 255 and g >= 0 and g <= 255 and b >= 0 and b <= 255 then
+				--values are valid 0..255, convert to 0..1
+				r = r / 255
+				g = g / 255
+				b = b / 255
+			else
+				--values are invalid
+				usererr(info, inputpos, format(L["'%s' - values must all be either in the range 0-1 or 0-255."], str))
+			end
+		end
+
+		do_final(info, inputpos, tab, "set", r,g,b,a)
+
+	elseif tab.type=="keybinding" then
+		------------ keybinding --------------------------------------------
+		local str = strtrim(strlower(str))
+		if str == "" then
+			--TODO: Show current value
+			return
+		end
+		local value = keybindingValidateFunc(str:upper())
+		if value == false then
+			usererr(info, inputpos, format(L["'%s' - Invalid Keybinding."], str))
+			return
+		end
+
+		do_final(info, inputpos, tab, "set", value)
+
+	elseif tab.type=="description" then
+		------------ description --------------------
+		-- ignore description, GUI config only
+	else
+		err(info, inputpos, "unknown options table item type '"..tostring(tab.type).."'")
+	end
+end
+
+--- Handle the chat command.
+-- This is usually called from a chat command handler to parse the command input as operations on an aceoptions table.\\
+-- AceConfigCmd uses this function internally when a slash command is registered with `:CreateChatCommand`
+-- @param slashcmd The slash command WITHOUT leading slash (only used for error output)
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param input The commandline input (as given by the WoW handler, i.e. without the command itself)
+-- @usage
+-- MyAddon = LibStub("AceAddon-3.0"):NewAddon("MyAddon", "AceConsole-3.0")
+-- -- Use AceConsole-3.0 to register a Chat Command
+-- MyAddon:RegisterChatCommand("mychat", "ChatCommand")
+--
+-- -- Show the GUI if no input is supplied, otherwise handle the chat input.
+-- function MyAddon:ChatCommand(input)
+--   -- Assuming "MyOptions" is the appName of a valid options table
+--   if not input or input:trim() == "" then
+--     LibStub("AceConfigDialog-3.0"):Open("MyOptions")
+--   else
+--     LibStub("AceConfigCmd-3.0").HandleCommand(MyAddon, "mychat", "MyOptions", input)
+--   end
+-- end
+function AceConfigCmd:HandleCommand(slashcmd, appName, input)
+
+	local optgetter = cfgreg:GetOptionsTable(appName)
+	if not optgetter then
+		error([[Usage: HandleCommand("slashcmd", "appName", "input"): 'appName' - no options table "]]..tostring(appName)..[[" has been registered]], 2)
+	end
+	local options = assert( optgetter("cmd", MAJOR) )
+
+	local info = {   -- Don't try to recycle this, it gets handed off to callbacks and whatnot
+		[0] = slashcmd,
+		appName = appName,
+		options = options,
+		input = input,
+		self = self,
+		handler = self,
+		uiType = "cmd",
+		uiName = MAJOR,
+	}
+
+	handle(info, 1, options, 0)  -- (info, inputpos, table, depth)
+end
+
+--- Utility function to create a slash command handler.
+-- Also registers tab completion with AceTab
+-- @param slashcmd The slash command WITHOUT leading slash (only used for error output)
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+function AceConfigCmd:CreateChatCommand(slashcmd, appName)
+	if not AceConsole then
+		AceConsole = LibStub(AceConsoleName)
+	end
+	if AceConsole.RegisterChatCommand(self, slashcmd, function(input)
+				AceConfigCmd.HandleCommand(self, slashcmd, appName, input)	-- upgradable
+		end,
+	true) then -- succesfully registered so lets get the command -> app table in
+		commands[slashcmd] = appName
+	end
+end
+
+--- Utility function that returns the options table that belongs to a slashcommand.
+-- Designed to be used for the AceTab interface.
+-- @param slashcmd The slash command WITHOUT leading slash (only used for error output)
+-- @return The options table associated with the slash command (or nil if the slash command was not registered)
+function AceConfigCmd:GetChatCommandOptions(slashcmd)
+	return commands[slashcmd]
+end
diff --git a/Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.xml b/Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.xml
new file mode 100755
index 0000000..c8caf34
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfigCmd-3.0/AceConfigCmd-3.0.xml
@@ -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="AceConfigCmd-3.0.lua"/>
+</Ui>
diff --git a/Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.lua b/Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.lua
new file mode 100755
index 0000000..c0309da
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.lua
@@ -0,0 +1,1983 @@
+--- AceConfigDialog-3.0 generates AceGUI-3.0 based windows based on option tables.
+-- @class file
+-- @name AceConfigDialog-3.0
+-- @release $Id: AceConfigDialog-3.0.lua 1169 2018-02-27 16:18:28Z nevcairiel $
+
+local LibStub = LibStub
+local gui = LibStub("AceGUI-3.0")
+local reg = LibStub("AceConfigRegistry-3.0")
+
+local MAJOR, MINOR = "AceConfigDialog-3.0", 66
+local AceConfigDialog, oldminor = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not AceConfigDialog then return end
+
+AceConfigDialog.OpenFrames = AceConfigDialog.OpenFrames or {}
+AceConfigDialog.Status = AceConfigDialog.Status or {}
+AceConfigDialog.frame = AceConfigDialog.frame or CreateFrame("Frame")
+
+AceConfigDialog.frame.apps = AceConfigDialog.frame.apps or {}
+AceConfigDialog.frame.closing = AceConfigDialog.frame.closing or {}
+AceConfigDialog.frame.closeAllOverride = AceConfigDialog.frame.closeAllOverride or {}
+
+-- Lua APIs
+local tconcat, tinsert, tsort, tremove, tsort = table.concat, table.insert, table.sort, table.remove, table.sort
+local strmatch, format = string.match, string.format
+local assert, loadstring, error = assert, loadstring, error
+local pairs, next, select, type, unpack, wipe, ipairs = pairs, next, select, type, unpack, wipe, ipairs
+local rawset, tostring, tonumber = rawset, tostring, tonumber
+local math_min, math_max, math_floor = math.min, math.max, math.floor
+
+-- Global vars/functions that we don't upvalue since they might get hooked, or upgraded
+-- List them here for Mikk's FindGlobals script
+-- GLOBALS: NORMAL_FONT_COLOR, GameTooltip, StaticPopupDialogs, ACCEPT, CANCEL, StaticPopup_Show
+-- GLOBALS: PlaySound, GameFontHighlight, GameFontHighlightSmall, GameFontHighlightLarge
+-- GLOBALS: CloseSpecialWindows, InterfaceOptions_AddCategory, geterrorhandler
+
+local emptyTbl = {}
+
+--[[
+	 xpcall safecall implementation
+]]
+local xpcall = xpcall
+
+local function errorhandler(err)
+	return geterrorhandler()(err)
+end
+
+local function CreateDispatcher(argCount)
+	local code = [[
+		local xpcall, eh = ...
+		local method, ARGS
+		local function call() return method(ARGS) end
+
+		local function dispatch(func, ...)
+			 method = func
+			 if not method then return end
+			 ARGS = ...
+			 return xpcall(call, eh)
+		end
+
+		return dispatch
+	]]
+
+	local ARGS = {}
+	for i = 1, argCount do ARGS[i] = "arg"..i end
+	code = code:gsub("ARGS", tconcat(ARGS, ", "))
+	return assert(loadstring(code, "safecall Dispatcher["..argCount.."]"))(xpcall, errorhandler)
+end
+
+local Dispatchers = setmetatable({}, {__index=function(self, argCount)
+	local dispatcher = CreateDispatcher(argCount)
+	rawset(self, argCount, dispatcher)
+	return dispatcher
+end})
+Dispatchers[0] = function(func)
+	return xpcall(func, errorhandler)
+end
+
+local function safecall(func, ...)
+	return Dispatchers[select("#", ...)](func, ...)
+end
+
+local width_multiplier = 170
+
+--[[
+Group Types
+  Tree 	- All Descendant Groups will all become nodes on the tree, direct child options will appear above the tree
+  		- Descendant Groups with inline=true and thier children will not become nodes
+
+  Tab	- Direct Child Groups will become tabs, direct child options will appear above the tab control
+  		- Grandchild groups will default to inline unless specified otherwise
+
+  Select- Same as Tab but with entries in a dropdown rather than tabs
+
+
+  Inline Groups
+  	- Will not become nodes of a select group, they will be effectivly part of thier parent group seperated by a border
+  	- If declared on a direct child of a root node of a select group, they will appear above the group container control
+  	- When a group is displayed inline, all descendants will also be inline members of the group
+
+]]
+
+-- Recycling functions
+local new, del, copy
+--newcount, delcount,createdcount,cached = 0,0,0
+do
+	local pool = setmetatable({},{__mode="k"})
+	function new()
+		--newcount = newcount + 1
+		local t = next(pool)
+		if t then
+			pool[t] = nil
+			return t
+		else
+			--createdcount = createdcount + 1
+			return {}
+		end
+	end
+	function copy(t)
+		local c = new()
+		for k, v in pairs(t) do
+			c[k] = v
+		end
+		return c
+	end
+	function del(t)
+		--delcount = delcount + 1
+		wipe(t)
+		pool[t] = true
+	end
+--	function cached()
+--		local n = 0
+--		for k in pairs(pool) do
+--			n = n + 1
+--		end
+--		return n
+--	end
+end
+
+-- picks the first non-nil value and returns it
+local function pickfirstset(...)
+  for i=1,select("#",...) do
+    if select(i,...)~=nil then
+      return select(i,...)
+    end
+  end
+end
+
+--gets an option from a given group, checking plugins
+local function GetSubOption(group, key)
+	if group.plugins then
+		for plugin, t in pairs(group.plugins) do
+			if t[key] then
+				return t[key]
+			end
+		end
+	end
+
+	return group.args[key]
+end
+
+--Option member type definitions, used to decide how to access it
+
+--Is the member Inherited from parent options
+local isInherited = {
+	set = true,
+	get = true,
+	func = true,
+	confirm = true,
+	validate = true,
+	disabled = true,
+	hidden = true
+}
+
+--Does a string type mean a literal value, instead of the default of a method of the handler
+local stringIsLiteral = {
+	name = true,
+	desc = true,
+	icon = true,
+	usage = true,
+	width = true,
+	image = true,
+	fontSize = true,
+}
+
+--Is Never a function or method
+local allIsLiteral = {
+	type = true,
+	descStyle = true,
+	imageWidth = true,
+	imageHeight = true,
+}
+
+--gets the value for a member that could be a function
+--function refs are called with an info arg
+--every other type is returned
+local function GetOptionsMemberValue(membername, option, options, path, appName, ...)
+	--get definition for the member
+	local inherits = isInherited[membername]
+
+
+	--get the member of the option, traversing the tree if it can be inherited
+	local member
+
+	if inherits then
+		local group = options
+		if group[membername] ~= nil then
+			member = group[membername]
+		end
+		for i = 1, #path do
+			group = GetSubOption(group, path[i])
+			if group[membername] ~= nil then
+				member = group[membername]
+			end
+		end
+	else
+		member = option[membername]
+	end
+
+	--check if we need to call a functon, or if we have a literal value
+	if ( not allIsLiteral[membername] ) and ( type(member) == "function" or ((not stringIsLiteral[membername]) and type(member) == "string") ) then
+		--We have a function to call
+		local info = new()
+		--traverse the options table, picking up the handler and filling the info with the path
+		local handler
+		local group = options
+		handler = group.handler or handler
+
+		for i = 1, #path do
+			group = GetSubOption(group, path[i])
+			info[i] = path[i]
+			handler = group.handler or handler
+		end
+
+		info.options = options
+		info.appName = appName
+		info[0] = appName
+		info.arg = option.arg
+		info.handler = handler
+		info.option = option
+		info.type = option.type
+		info.uiType = "dialog"
+		info.uiName = MAJOR
+
+		local a, b, c ,d
+		--using 4 returns for the get of a color type, increase if a type needs more
+		if type(member) == "function" then
+			--Call the function
+			a,b,c,d = member(info, ...)
+		else
+			--Call the method
+			if handler and handler[member] then
+				a,b,c,d = handler[member](handler, info, ...)
+			else
+				error(format("Method %s doesn't exist in handler for type %s", member, membername))
+			end
+		end
+		del(info)
+		return a,b,c,d
+	else
+		--The value isnt a function to call, return it
+		return member
+	end
+end
+
+--[[calls an options function that could be inherited, method name or function ref
+local function CallOptionsFunction(funcname ,option, options, path, appName, ...)
+	local info = new()
+
+	local func
+	local group = options
+	local handler
+
+	--build the info table containing the path
+	-- pick up functions while traversing the tree
+	if group[funcname] ~= nil then
+		func = group[funcname]
+	end
+	handler = group.handler or handler
+
+	for i, v in ipairs(path) do
+		group = GetSubOption(group, v)
+		info[i] = v
+		if group[funcname] ~= nil then
+			func =  group[funcname]
+		end
+		handler = group.handler or handler
+	end
+
+	info.options = options
+	info[0] = appName
+	info.arg = option.arg
+
+	local a, b, c ,d
+	if type(func) == "string" then
+		if handler and handler[func] then
+			a,b,c,d = handler[func](handler, info, ...)
+		else
+			error(string.format("Method %s doesn't exist in handler for type func", func))
+		end
+	elseif type(func) == "function" then
+		a,b,c,d = func(info, ...)
+	end
+	del(info)
+	return a,b,c,d
+end
+--]]
+
+--tables to hold orders and names for options being sorted, will be created with new()
+--prevents needing to call functions repeatedly while sorting
+local tempOrders
+local tempNames
+
+local function compareOptions(a,b)
+	if not a then
+		return true
+	end
+	if not b then
+		return false
+	end
+	local OrderA, OrderB = tempOrders[a] or 100, tempOrders[b] or 100
+	if OrderA == OrderB then
+		local NameA = (type(tempNames[a]) == "string") and tempNames[a] or ""
+		local NameB = (type(tempNames[b]) == "string") and tempNames[b] or ""
+		return NameA:upper() < NameB:upper()
+	end
+	if OrderA < 0 then
+		if OrderB > 0 then
+			return false
+		end
+	else
+		if OrderB < 0 then
+			return true
+		end
+	end
+	return OrderA < OrderB
+end
+
+
+
+--builds 2 tables out of an options group
+-- keySort, sorted keys
+-- opts, combined options from .plugins and args
+local function BuildSortedOptionsTable(group, keySort, opts, options, path, appName)
+	tempOrders = new()
+	tempNames = new()
+
+	if group.plugins then
+		for plugin, t in pairs(group.plugins) do
+			for k, v in pairs(t) do
+				if not opts[k] then
+					tinsert(keySort, k)
+					opts[k] = v
+
+					path[#path+1] = k
+					tempOrders[k] = GetOptionsMemberValue("order", v, options, path, appName)
+					tempNames[k] = GetOptionsMemberValue("name", v, options, path, appName)
+					path[#path] = nil
+				end
+			end
+		end
+	end
+
+	for k, v in pairs(group.args) do
+		if not opts[k] then
+			tinsert(keySort, k)
+			opts[k] = v
+
+			path[#path+1] = k
+			tempOrders[k] = GetOptionsMemberValue("order", v, options, path, appName)
+			tempNames[k] = GetOptionsMemberValue("name", v, options, path, appName)
+			path[#path] = nil
+		end
+	end
+
+	tsort(keySort, compareOptions)
+
+	del(tempOrders)
+	del(tempNames)
+end
+
+local function DelTree(tree)
+	if tree.children then
+		local childs = tree.children
+		for i = 1, #childs do
+			DelTree(childs[i])
+			del(childs[i])
+		end
+		del(childs)
+	end
+end
+
+local function CleanUserData(widget, event)
+
+	local user = widget:GetUserDataTable()
+
+	if user.path then
+		del(user.path)
+	end
+
+	if widget.type == "TreeGroup" then
+		local tree = user.tree
+		widget:SetTree(nil)
+		if tree then
+			for i = 1, #tree do
+				DelTree(tree[i])
+				del(tree[i])
+			end
+			del(tree)
+		end
+	end
+
+	if widget.type == "TabGroup" then
+		widget:SetTabs(nil)
+		if user.tablist then
+			del(user.tablist)
+		end
+	end
+
+	if widget.type == "DropdownGroup" then
+		widget:SetGroupList(nil)
+		if user.grouplist then
+			del(user.grouplist)
+		end
+		if user.orderlist then
+			del(user.orderlist)
+		end
+	end
+end
+
+-- - Gets a status table for the given appname and options path.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param path The path to the options (a table with all group keys)
+-- @return
+function AceConfigDialog:GetStatusTable(appName, path)
+	local status = self.Status
+
+	if not status[appName] then
+		status[appName] = {}
+		status[appName].status = {}
+		status[appName].children = {}
+	end
+
+	status = status[appName]
+
+	if path then
+		for i = 1, #path do
+			local v = path[i]
+			if not status.children[v] then
+				status.children[v] = {}
+				status.children[v].status = {}
+				status.children[v].children = {}
+			end
+			status = status.children[v]
+		end
+	end
+
+	return status.status
+end
+
+--- Selects the specified path in the options window.
+-- The path specified has to match the keys of the groups in the table.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param ... The path to the key that should be selected
+function AceConfigDialog:SelectGroup(appName, ...)
+	local path = new()
+
+
+	local app = reg:GetOptionsTable(appName)
+	if not app then
+		error(("%s isn't registed with AceConfigRegistry, unable to open config"):format(appName), 2)
+	end
+	local options = app("dialog", MAJOR)
+	local group = options
+	local status = self:GetStatusTable(appName, path)
+	if not status.groups then
+		status.groups = {}
+	end
+	status = status.groups
+	local treevalue
+	local treestatus
+
+	for n = 1, select("#",...) do
+		local key = select(n, ...)
+
+		if group.childGroups == "tab" or group.childGroups == "select" then
+			--if this is a tab or select group, select the group
+			status.selected = key
+			--children of this group are no longer extra levels of a tree
+			treevalue = nil
+		else
+			--tree group by default
+			if treevalue then
+				--this is an extra level of a tree group, build a uniquevalue for it
+				treevalue = treevalue.."\001"..key
+			else
+				--this is the top level of a tree group, the uniquevalue is the same as the key
+				treevalue = key
+				if not status.groups then
+					status.groups = {}
+				end
+				--save this trees status table for any extra levels or groups
+				treestatus = status
+			end
+			--make sure that the tree entry is open, and select it.
+			--the selected group will be overwritten if a child is the final target but still needs to be open
+			treestatus.selected = treevalue
+			treestatus.groups[treevalue] = true
+
+		end
+
+		--move to the next group in the path
+		group = GetSubOption(group, key)
+		if not group then
+			break
+		end
+		tinsert(path, key)
+		status = self:GetStatusTable(appName, path)
+		if not status.groups then
+			status.groups = {}
+		end
+		status = status.groups
+	end
+
+	del(path)
+	reg:NotifyChange(appName)
+end
+
+local function OptionOnMouseOver(widget, event)
+	--show a tooltip/set the status bar to the desc text
+	local user = widget:GetUserDataTable()
+	local opt = user.option
+	local options = user.options
+	local path = user.path
+	local appName = user.appName
+
+	GameTooltip:SetOwner(widget.frame, "ANCHOR_TOPRIGHT")
+	local name = GetOptionsMemberValue("name", opt, options, path, appName)
+	local desc = GetOptionsMemberValue("desc", opt, options, path, appName)
+	local usage = GetOptionsMemberValue("usage", opt, options, path, appName)
+	local descStyle = opt.descStyle
+
+	if descStyle and descStyle ~= "tooltip" then return end
+
+	GameTooltip:SetText(name, 1, .82, 0, true)
+
+	if opt.type == "multiselect" then
+		GameTooltip:AddLine(user.text, 0.5, 0.5, 0.8, true)
+	end
+	if type(desc) == "string" then
+		GameTooltip:AddLine(desc, 1, 1, 1, true)
+	end
+	if type(usage) == "string" then
+		GameTooltip:AddLine("Usage: "..usage, NORMAL_FONT_COLOR.r, NORMAL_FONT_COLOR.g, NORMAL_FONT_COLOR.b, true)
+	end
+
+	GameTooltip:Show()
+end
+
+local function OptionOnMouseLeave(widget, event)
+	GameTooltip:Hide()
+end
+
+local function GetFuncName(option)
+	local type = option.type
+	if type == "execute" then
+		return "func"
+	else
+		return "set"
+	end
+end
+local function confirmPopup(appName, rootframe, basepath, info, message, func, ...)
+	if not StaticPopupDialogs["ACECONFIGDIALOG30_CONFIRM_DIALOG"] then
+		StaticPopupDialogs["ACECONFIGDIALOG30_CONFIRM_DIALOG"] = {}
+	end
+	local t = StaticPopupDialogs["ACECONFIGDIALOG30_CONFIRM_DIALOG"]
+	for k in pairs(t) do
+		t[k] = nil
+	end
+	t.text = message
+	t.button1 = ACCEPT
+	t.button2 = CANCEL
+	t.preferredIndex = STATICPOPUP_NUMDIALOGS
+	local dialog, oldstrata
+	t.OnAccept = function()
+		safecall(func, unpack(t))
+		if dialog and oldstrata then
+			dialog:SetFrameStrata(oldstrata)
+		end
+		AceConfigDialog:Open(appName, rootframe, unpack(basepath or emptyTbl))
+		del(info)
+	end
+	t.OnCancel = function()
+		if dialog and oldstrata then
+			dialog:SetFrameStrata(oldstrata)
+		end
+		AceConfigDialog:Open(appName, rootframe, unpack(basepath or emptyTbl))
+		del(info)
+	end
+	for i = 1, select("#", ...) do
+		t[i] = select(i, ...) or false
+	end
+	t.timeout = 0
+	t.whileDead = 1
+	t.hideOnEscape = 1
+
+	dialog = StaticPopup_Show("ACECONFIGDIALOG30_CONFIRM_DIALOG")
+	if dialog then
+		oldstrata = dialog:GetFrameStrata()
+		dialog:SetFrameStrata("TOOLTIP")
+	end
+end
+
+local function validationErrorPopup(message)
+	if not StaticPopupDialogs["ACECONFIGDIALOG30_VALIDATION_ERROR_DIALOG"] then
+		StaticPopupDialogs["ACECONFIGDIALOG30_VALIDATION_ERROR_DIALOG"] = {}
+	end
+	local t = StaticPopupDialogs["ACECONFIGDIALOG30_VALIDATION_ERROR_DIALOG"]
+	t.text = message
+	t.button1 = OKAY
+	t.preferredIndex = STATICPOPUP_NUMDIALOGS
+	local dialog, oldstrata
+	t.OnAccept = function()
+		if dialog and oldstrata then
+			dialog:SetFrameStrata(oldstrata)
+		end
+	end
+	t.timeout = 0
+	t.whileDead = 1
+	t.hideOnEscape = 1
+
+	dialog = StaticPopup_Show("ACECONFIGDIALOG30_VALIDATION_ERROR_DIALOG")
+	if dialog then
+		oldstrata = dialog:GetFrameStrata()
+		dialog:SetFrameStrata("TOOLTIP")
+	end
+end
+
+local function ActivateControl(widget, event, ...)
+	--This function will call the set / execute handler for the widget
+	--widget:GetUserDataTable() contains the needed info
+	local user = widget:GetUserDataTable()
+	local option = user.option
+	local options = user.options
+	local path = user.path
+	local info = new()
+
+	local func
+	local group = options
+	local funcname = GetFuncName(option)
+	local handler
+	local confirm
+	local validate
+	--build the info table containing the path
+	-- pick up functions while traversing the tree
+	if group[funcname] ~= nil then
+		func =  group[funcname]
+	end
+	handler = group.handler or handler
+	confirm = group.confirm
+	validate = group.validate
+	for i = 1, #path do
+		local v = path[i]
+		group = GetSubOption(group, v)
+		info[i] = v
+		if group[funcname] ~= nil then
+			func =  group[funcname]
+		end
+		handler = group.handler or handler
+		if group.confirm ~= nil then
+			confirm = group.confirm
+		end
+		if group.validate ~= nil then
+			validate = group.validate
+		end
+	end
+
+	info.options = options
+	info.appName = user.appName
+	info.arg = option.arg
+	info.handler = handler
+	info.option = option
+	info.type = option.type
+	info.uiType = "dialog"
+	info.uiName = MAJOR
+
+	local name
+	if type(option.name) == "function" then
+		name = option.name(info)
+	elseif type(option.name) == "string" then
+		name = option.name
+	else
+		name = ""
+	end
+	local usage = option.usage
+	local pattern = option.pattern
+
+	local validated = true
+
+	if option.type == "input" then
+		if type(pattern)=="string" then
+			if not strmatch(..., pattern) then
+				validated = false
+			end
+		end
+	end
+
+	local success
+	if validated and option.type ~= "execute" then
+		if type(validate) == "string" then
+			if handler and handler[validate] then
+				success, validated = safecall(handler[validate], handler, info, ...)
+				if not success then validated = false end
+			else
+				error(format("Method %s doesn't exist in handler for type execute", validate))
+			end
+		elseif type(validate) == "function" then
+			success, validated = safecall(validate, info, ...)
+			if not success then validated = false end
+		end
+	end
+
+	local rootframe = user.rootframe
+	if not validated or type(validated) == "string" then
+		if not validated then
+			if usage then
+				validated = name..": "..usage
+			else
+				if pattern then
+					validated = name..": Expected "..pattern
+				else
+					validated = name..": Invalid Value"
+				end
+			end
+		end
+
+		-- show validate message
+		if rootframe.SetStatusText then
+			rootframe:SetStatusText(validated)
+		else
+			validationErrorPopup(validated)
+		end
+		PlaySound(882) -- SOUNDKIT.IG_PLAYER_INVITE_DECLINE || _DECLINE is actually missing from the table
+		del(info)
+		return true
+	else
+
+		local confirmText = option.confirmText
+		--call confirm func/method
+		if type(confirm) == "string" then
+			if handler and handler[confirm] then
+				success, confirm = safecall(handler[confirm], handler, info, ...)
+				if success and type(confirm) == "string" then
+					confirmText = confirm
+					confirm = true
+				elseif not success then
+					confirm = false
+				end
+			else
+				error(format("Method %s doesn't exist in handler for type confirm", confirm))
+			end
+		elseif type(confirm) == "function" then
+			success, confirm = safecall(confirm, info, ...)
+			if success and type(confirm) == "string" then
+				confirmText = confirm
+				confirm = true
+			elseif not success then
+				confirm = false
+			end
+		end
+
+		--confirm if needed
+		if type(confirm) == "boolean" then
+			if confirm then
+				if not confirmText then
+					local name, desc = option.name, option.desc
+					if type(name) == "function" then
+						name = name(info)
+					end
+					if type(desc) == "function" then
+						desc = desc(info)
+					end
+					confirmText = name
+					if desc then
+						confirmText = confirmText.." - "..desc
+					end
+				end
+
+				local iscustom = user.rootframe:GetUserData("iscustom")
+				local rootframe
+
+				if iscustom then
+					rootframe = user.rootframe
+				end
+				local basepath = user.rootframe:GetUserData("basepath")
+				if type(func) == "string" then
+					if handler and handler[func] then
+						confirmPopup(user.appName, rootframe, basepath, info, confirmText, handler[func], handler, info, ...)
+					else
+						error(format("Method %s doesn't exist in handler for type func", func))
+					end
+				elseif type(func) == "function" then
+					confirmPopup(user.appName, rootframe, basepath, info, confirmText, func, info, ...)
+				end
+				--func will be called and info deleted when the confirm dialog is responded to
+				return
+			end
+		end
+
+		--call the function
+		if type(func) == "string" then
+			if handler and handler[func] then
+				safecall(handler[func],handler, info, ...)
+			else
+				error(format("Method %s doesn't exist in handler for type func", func))
+			end
+		elseif type(func) == "function" then
+			safecall(func,info, ...)
+		end
+
+
+
+		local iscustom = user.rootframe:GetUserData("iscustom")
+		local basepath = user.rootframe:GetUserData("basepath") or emptyTbl
+		--full refresh of the frame, some controls dont cause this on all events
+		if option.type == "color" then
+			if event == "OnValueConfirmed" then
+
+				if iscustom then
+					AceConfigDialog:Open(user.appName, user.rootframe, unpack(basepath))
+				else
+					AceConfigDialog:Open(user.appName, unpack(basepath))
+				end
+			end
+		elseif option.type == "range" then
+			if event == "OnMouseUp" then
+				if iscustom then
+					AceConfigDialog:Open(user.appName, user.rootframe, unpack(basepath))
+				else
+					AceConfigDialog:Open(user.appName, unpack(basepath))
+				end
+			end
+		--multiselects don't cause a refresh on 'OnValueChanged' only 'OnClosed'
+		elseif option.type == "multiselect" then
+			user.valuechanged = true
+		else
+			if iscustom then
+				AceConfigDialog:Open(user.appName, user.rootframe, unpack(basepath))
+			else
+				AceConfigDialog:Open(user.appName, unpack(basepath))
+			end
+		end
+
+	end
+	del(info)
+end
+
+local function ActivateSlider(widget, event, value)
+	local option = widget:GetUserData("option")
+	local min, max, step = option.min or (not option.softMin and 0 or nil), option.max or (not option.softMax and 100 or nil), option.step
+	if min then
+		if step then
+			value = math_floor((value - min) / step + 0.5) * step + min
+		end
+		value = math_max(value, min)
+	end
+	if max then
+		value = math_min(value, max)
+	end
+	ActivateControl(widget,event,value)
+end
+
+--called from a checkbox that is part of an internally created multiselect group
+--this type is safe to refresh on activation of one control
+local function ActivateMultiControl(widget, event, ...)
+	ActivateControl(widget, event, widget:GetUserData("value"), ...)
+	local user = widget:GetUserDataTable()
+	local iscustom = user.rootframe:GetUserData("iscustom")
+	local basepath = user.rootframe:GetUserData("basepath") or emptyTbl
+	if iscustom then
+		AceConfigDialog:Open(user.appName, user.rootframe, unpack(basepath))
+	else
+		AceConfigDialog:Open(user.appName, unpack(basepath))
+	end
+end
+
+local function MultiControlOnClosed(widget, event, ...)
+	local user = widget:GetUserDataTable()
+	if user.valuechanged then
+		local iscustom = user.rootframe:GetUserData("iscustom")
+		local basepath = user.rootframe:GetUserData("basepath") or emptyTbl
+		if iscustom then
+			AceConfigDialog:Open(user.appName, user.rootframe, unpack(basepath))
+		else
+			AceConfigDialog:Open(user.appName, unpack(basepath))
+		end
+	end
+end
+
+local function FrameOnClose(widget, event)
+	local appName = widget:GetUserData("appName")
+	AceConfigDialog.OpenFrames[appName] = nil
+	gui:Release(widget)
+end
+
+local function CheckOptionHidden(option, options, path, appName)
+	--check for a specific boolean option
+	local hidden = pickfirstset(option.dialogHidden,option.guiHidden)
+	if hidden ~= nil then
+		return hidden
+	end
+
+	return GetOptionsMemberValue("hidden", option, options, path, appName)
+end
+
+local function CheckOptionDisabled(option, options, path, appName)
+	--check for a specific boolean option
+	local disabled = pickfirstset(option.dialogDisabled,option.guiDisabled)
+	if disabled ~= nil then
+		return disabled
+	end
+
+	return GetOptionsMemberValue("disabled", option, options, path, appName)
+end
+--[[
+local function BuildTabs(group, options, path, appName)
+	local tabs = new()
+	local text = new()
+	local keySort = new()
+	local opts = new()
+
+	BuildSortedOptionsTable(group, keySort, opts, options, path, appName)
+
+	for i = 1, #keySort do
+		local k = keySort[i]
+		local v = opts[k]
+		if v.type == "group" then
+			path[#path+1] = k
+			local inline = pickfirstset(v.dialogInline,v.guiInline,v.inline, false)
+			local hidden = CheckOptionHidden(v, options, path, appName)
+			if not inline and not hidden then
+				tinsert(tabs, k)
+				text[k] = GetOptionsMemberValue("name", v, options, path, appName)
+			end
+			path[#path] = nil
+		end
+	end
+
+	del(keySort)
+	del(opts)
+
+	return tabs, text
+end
+]]
+local function BuildSelect(group, options, path, appName)
+	local groups = new()
+	local order = new()
+	local keySort = new()
+	local opts = new()
+
+	BuildSortedOptionsTable(group, keySort, opts, options, path, appName)
+
+	for i = 1, #keySort do
+		local k = keySort[i]
+		local v = opts[k]
+		if v.type == "group" then
+			path[#path+1] = k
+			local inline = pickfirstset(v.dialogInline,v.guiInline,v.inline, false)
+			local hidden = CheckOptionHidden(v, options, path, appName)
+			if not inline and not hidden then
+				groups[k] = GetOptionsMemberValue("name", v, options, path, appName)
+				tinsert(order, k)
+			end
+			path[#path] = nil
+		end
+	end
+
+	del(opts)
+	del(keySort)
+
+	return groups, order
+end
+
+local function BuildSubGroups(group, tree, options, path, appName)
+	local keySort = new()
+	local opts = new()
+
+	BuildSortedOptionsTable(group, keySort, opts, options, path, appName)
+
+	for i = 1, #keySort do
+		local k = keySort[i]
+		local v = opts[k]
+		if v.type == "group" then
+			path[#path+1] = k
+			local inline = pickfirstset(v.dialogInline,v.guiInline,v.inline, false)
+			local hidden = CheckOptionHidden(v, options, path, appName)
+			if not inline and not hidden then
+				local entry = new()
+				entry.value = k
+				entry.text = GetOptionsMemberValue("name", v, options, path, appName)
+				entry.icon = GetOptionsMemberValue("icon", v, options, path, appName)
+				entry.iconCoords = GetOptionsMemberValue("iconCoords", v, options, path, appName)
+				entry.disabled = CheckOptionDisabled(v, options, path, appName)
+				if not tree.children then tree.children = new() end
+				tinsert(tree.children,entry)
+				if (v.childGroups or "tree") == "tree" then
+					BuildSubGroups(v,entry, options, path, appName)
+				end
+			end
+			path[#path] = nil
+		end
+	end
+
+	del(keySort)
+	del(opts)
+end
+
+local function BuildGroups(group, options, path, appName, recurse)
+	local tree = new()
+	local keySort = new()
+	local opts = new()
+
+	BuildSortedOptionsTable(group, keySort, opts, options, path, appName)
+
+	for i = 1, #keySort do
+		local k = keySort[i]
+		local v = opts[k]
+		if v.type == "group" then
+			path[#path+1] = k
+			local inline = pickfirstset(v.dialogInline,v.guiInline,v.inline, false)
+			local hidden = CheckOptionHidden(v, options, path, appName)
+			if not inline and not hidden then
+				local entry = new()
+				entry.value = k
+				entry.text = GetOptionsMemberValue("name", v, options, path, appName)
+				entry.icon = GetOptionsMemberValue("icon", v, options, path, appName)
+				entry.iconCoords = GetOptionsMemberValue("iconCoords", v, options, path, appName)
+				entry.disabled = CheckOptionDisabled(v, options, path, appName)
+				tinsert(tree,entry)
+				if recurse and (v.childGroups or "tree") == "tree" then
+					BuildSubGroups(v,entry, options, path, appName)
+				end
+			end
+			path[#path] = nil
+		end
+	end
+	del(keySort)
+	del(opts)
+	return tree
+end
+
+local function InjectInfo(control, options, option, path, rootframe, appName)
+	local user = control:GetUserDataTable()
+	for i = 1, #path do
+		user[i] = path[i]
+	end
+	user.rootframe = rootframe
+	user.option = option
+	user.options = options
+	user.path = copy(path)
+	user.appName = appName
+	control:SetCallback("OnRelease", CleanUserData)
+	control:SetCallback("OnLeave", OptionOnMouseLeave)
+	control:SetCallback("OnEnter", OptionOnMouseOver)
+end
+
+
+--[[
+	options - root of the options table being fed
+	container - widget that controls will be placed in
+	rootframe - Frame object the options are in
+	path - table with the keys to get to the group being fed
+--]]
+
+local function FeedOptions(appName, options,container,rootframe,path,group,inline)
+	local keySort = new()
+	local opts = new()
+
+	BuildSortedOptionsTable(group, keySort, opts, options, path, appName)
+
+	for i = 1, #keySort do
+		local k = keySort[i]
+		local v = opts[k]
+		tinsert(path, k)
+		local hidden = CheckOptionHidden(v, options, path, appName)
+		local name = GetOptionsMemberValue("name", v, options, path, appName)
+		if not hidden then
+			if v.type == "group" then
+				if inline or pickfirstset(v.dialogInline,v.guiInline,v.inline, false) then
+					--Inline group
+					local GroupContainer
+					if name and name ~= "" then
+						GroupContainer = gui:Create("InlineGroup")
+						GroupContainer:SetTitle(name or "")
+					else
+						GroupContainer = gui:Create("SimpleGroup")
+					end
+
+					GroupContainer.width = "fill"
+					GroupContainer:SetLayout("flow")
+					container:AddChild(GroupContainer)
+					FeedOptions(appName,options,GroupContainer,rootframe,path,v,true)
+				end
+			else
+				--Control to feed
+				local control
+
+				local name = GetOptionsMemberValue("name", v, options, path, appName)
+
+				if v.type == "execute" then
+
+					local imageCoords = GetOptionsMemberValue("imageCoords",v, options, path, appName)
+					local image, width, height = GetOptionsMemberValue("image",v, options, path, appName)
+
+					if type(image) == "string" or type(image) == "number" then
+						control = gui:Create("Icon")
+						if not width then
+							width = GetOptionsMemberValue("imageWidth",v, options, path, appName)
+						end
+						if not height then
+							height = GetOptionsMemberValue("imageHeight",v, options, path, appName)
+						end
+						if type(imageCoords) == "table" then
+							control:SetImage(image, unpack(imageCoords))
+						else
+							control:SetImage(image)
+						end
+						if type(width) ~= "number" then
+							width = 32
+						end
+						if type(height) ~= "number" then
+							height = 32
+						end
+						control:SetImageSize(width, height)
+						control:SetLabel(name)
+					else
+						control = gui:Create("Button")
+						control:SetText(name)
+					end
+					control:SetCallback("OnClick",ActivateControl)
+
+				elseif v.type == "input" then
+					local controlType = v.dialogControl or v.control or (v.multiline and "MultiLineEditBox") or "EditBox"
+					control = gui:Create(controlType)
+					if not control then
+						geterrorhandler()(("Invalid Custom Control Type - %s"):format(tostring(controlType)))
+						control = gui:Create(v.multiline and "MultiLineEditBox" or "EditBox")
+					end
+
+					if v.multiline and control.SetNumLines then
+						control:SetNumLines(tonumber(v.multiline) or 4)
+					end
+					control:SetLabel(name)
+					control:SetCallback("OnEnterPressed",ActivateControl)
+					local text = GetOptionsMemberValue("get",v, options, path, appName)
+					if type(text) ~= "string" then
+						text = ""
+					end
+					control:SetText(text)
+
+				elseif v.type == "toggle" then
+					control = gui:Create("CheckBox")
+					control:SetLabel(name)
+					control:SetTriState(v.tristate)
+					local value = GetOptionsMemberValue("get",v, options, path, appName)
+					control:SetValue(value)
+					control:SetCallback("OnValueChanged",ActivateControl)
+
+					if v.descStyle == "inline" then
+						local desc = GetOptionsMemberValue("desc", v, options, path, appName)
+						control:SetDescription(desc)
+					end
+
+					local image = GetOptionsMemberValue("image", v, options, path, appName)
+					local imageCoords = GetOptionsMemberValue("imageCoords", v, options, path, appName)
+
+					if type(image) == "string" or type(image) == "number" then
+						if type(imageCoords) == "table" then
+							control:SetImage(image, unpack(imageCoords))
+						else
+							control:SetImage(image)
+						end
+					end
+				elseif v.type == "range" then
+					control = gui:Create("Slider")
+					control:SetLabel(name)
+					control:SetSliderValues(v.softMin or v.min or 0, v.softMax or v.max or 100, v.bigStep or v.step or 0)
+					control:SetIsPercent(v.isPercent)
+					local value = GetOptionsMemberValue("get",v, options, path, appName)
+					if type(value) ~= "number" then
+						value = 0
+					end
+					control:SetValue(value)
+					control:SetCallback("OnValueChanged",ActivateSlider)
+					control:SetCallback("OnMouseUp",ActivateSlider)
+
+				elseif v.type == "select" then
+					local values = GetOptionsMemberValue("values", v, options, path, appName)
+					if v.style == "radio" then
+						local disabled = CheckOptionDisabled(v, options, path, appName)
+						local width = GetOptionsMemberValue("width",v,options,path,appName)
+						control = gui:Create("InlineGroup")
+						control:SetLayout("Flow")
+						control:SetTitle(name)
+						control.width = "fill"
+
+						control:PauseLayout()
+						local optionValue = GetOptionsMemberValue("get",v, options, path, appName)
+						local t = {}
+						for value, text in pairs(values) do
+							t[#t+1]=value
+						end
+						tsort(t)
+						for k, value in ipairs(t) do
+							local text = values[value]
+							local radio = gui:Create("CheckBox")
+							radio:SetLabel(text)
+							radio:SetUserData("value", value)
+							radio:SetUserData("text", text)
+							radio:SetDisabled(disabled)
+							radio:SetType("radio")
+							radio:SetValue(optionValue == value)
+							radio:SetCallback("OnValueChanged", ActivateMultiControl)
+							InjectInfo(radio, options, v, path, rootframe, appName)
+							control:AddChild(radio)
+							if width == "double" then
+								radio:SetWidth(width_multiplier * 2)
+							elseif width == "half" then
+								radio:SetWidth(width_multiplier / 2)
+							elseif (type(width) == "number") then
+								radio:SetWidth(width_multiplier * width)
+							elseif width == "full" then
+								radio.width = "fill"
+							else
+								radio:SetWidth(width_multiplier)
+							end
+						end
+						control:ResumeLayout()
+						control:DoLayout()
+					else
+						local controlType = v.dialogControl or v.control or "Dropdown"
+						control = gui:Create(controlType)
+						if not control then
+							geterrorhandler()(("Invalid Custom Control Type - %s"):format(tostring(controlType)))
+							control = gui:Create("Dropdown")
+						end
+						local itemType = v.itemControl
+						if itemType and not gui:GetWidgetVersion(itemType) then
+							geterrorhandler()(("Invalid Custom Item Type - %s"):format(tostring(itemType)))
+							itemType = nil
+						end
+						control:SetLabel(name)
+						control:SetList(values, nil, itemType)
+						local value = GetOptionsMemberValue("get",v, options, path, appName)
+						if not values[value] then
+							value = nil
+						end
+						control:SetValue(value)
+						control:SetCallback("OnValueChanged", ActivateControl)
+					end
+
+				elseif v.type == "multiselect" then
+					local values = GetOptionsMemberValue("values", v, options, path, appName)
+					local disabled = CheckOptionDisabled(v, options, path, appName)
+
+					local controlType = v.dialogControl or v.control
+
+					local valuesort = new()
+					if values then
+						for value, text in pairs(values) do
+							tinsert(valuesort, value)
+						end
+					end
+					tsort(valuesort)
+
+					if controlType then
+						control = gui:Create(controlType)
+						if not control then
+							geterrorhandler()(("Invalid Custom Control Type - %s"):format(tostring(controlType)))
+						end
+					end
+					if control then
+						control:SetMultiselect(true)
+						control:SetLabel(name)
+						control:SetList(values)
+						control:SetDisabled(disabled)
+						control:SetCallback("OnValueChanged",ActivateControl)
+						control:SetCallback("OnClosed", MultiControlOnClosed)
+						local width = GetOptionsMemberValue("width",v,options,path,appName)
+						if width == "double" then
+							control:SetWidth(width_multiplier * 2)
+						elseif width == "half" then
+							control:SetWidth(width_multiplier / 2)
+						elseif (type(width) == "number") then
+							control:SetWidth(width_multiplier * width)
+						elseif width == "full" then
+							control.width = "fill"
+						else
+							control:SetWidth(width_multiplier)
+						end
+						--check:SetTriState(v.tristate)
+						for i = 1, #valuesort do
+							local key = valuesort[i]
+							local value = GetOptionsMemberValue("get",v, options, path, appName, key)
+							control:SetItemValue(key,value)
+						end
+					else
+						control = gui:Create("InlineGroup")
+						control:SetLayout("Flow")
+						control:SetTitle(name)
+						control.width = "fill"
+
+						control:PauseLayout()
+						local width = GetOptionsMemberValue("width",v,options,path,appName)
+						for i = 1, #valuesort do
+							local value = valuesort[i]
+							local text = values[value]
+							local check = gui:Create("CheckBox")
+							check:SetLabel(text)
+							check:SetUserData("value", value)
+							check:SetUserData("text", text)
+							check:SetDisabled(disabled)
+							check:SetTriState(v.tristate)
+							check:SetValue(GetOptionsMemberValue("get",v, options, path, appName, value))
+							check:SetCallback("OnValueChanged",ActivateMultiControl)
+							InjectInfo(check, options, v, path, rootframe, appName)
+							control:AddChild(check)
+							if width == "double" then
+								check:SetWidth(width_multiplier * 2)
+							elseif width == "half" then
+								check:SetWidth(width_multiplier / 2)
+							elseif (type(width) == "number") then
+								control:SetWidth(width_multiplier * width)
+							elseif width == "full" then
+								check.width = "fill"
+							else
+								check:SetWidth(width_multiplier)
+							end
+						end
+						control:ResumeLayout()
+						control:DoLayout()
+
+
+					end
+
+					del(valuesort)
+
+				elseif v.type == "color" then
+					control = gui:Create("ColorPicker")
+					control:SetLabel(name)
+					control:SetHasAlpha(GetOptionsMemberValue("hasAlpha",v, options, path, appName))
+					control:SetColor(GetOptionsMemberValue("get",v, options, path, appName))
+					control:SetCallback("OnValueChanged",ActivateControl)
+					control:SetCallback("OnValueConfirmed",ActivateControl)
+
+				elseif v.type == "keybinding" then
+					control = gui:Create("Keybinding")
+					control:SetLabel(name)
+					control:SetKey(GetOptionsMemberValue("get",v, options, path, appName))
+					control:SetCallback("OnKeyChanged",ActivateControl)
+
+				elseif v.type == "header" then
+					control = gui:Create("Heading")
+					control:SetText(name)
+					control.width = "fill"
+
+				elseif v.type == "description" then
+					control = gui:Create("Label")
+					control:SetText(name)
+
+					local fontSize = GetOptionsMemberValue("fontSize",v, options, path, appName)
+					if fontSize == "medium" then
+						control:SetFontObject(GameFontHighlight)
+					elseif fontSize == "large" then
+						control:SetFontObject(GameFontHighlightLarge)
+					else -- small or invalid
+						control:SetFontObject(GameFontHighlightSmall)
+					end
+
+					local imageCoords = GetOptionsMemberValue("imageCoords",v, options, path, appName)
+					local image, width, height = GetOptionsMemberValue("image",v, options, path, appName)
+
+					if type(image) == "string" or type(image) == "number" then
+						if not width then
+							width = GetOptionsMemberValue("imageWidth",v, options, path, appName)
+						end
+						if not height then
+							height = GetOptionsMemberValue("imageHeight",v, options, path, appName)
+						end
+						if type(imageCoords) == "table" then
+							control:SetImage(image, unpack(imageCoords))
+						else
+							control:SetImage(image)
+						end
+						if type(width) ~= "number" then
+							width = 32
+						end
+						if type(height) ~= "number" then
+							height = 32
+						end
+						control:SetImageSize(width, height)
+					end
+					local width = GetOptionsMemberValue("width",v,options,path,appName)
+					control.width = not width and "fill"
+				end
+
+				--Common Init
+				if control then
+					if control.width ~= "fill" then
+						local width = GetOptionsMemberValue("width",v,options,path,appName)
+						if width == "double" then
+							control:SetWidth(width_multiplier * 2)
+						elseif width == "half" then
+							control:SetWidth(width_multiplier / 2)
+						elseif (type(width) == "number") then
+							control:SetWidth(width_multiplier * width)
+						elseif width == "full" then
+							control.width = "fill"
+						else
+							control:SetWidth(width_multiplier)
+						end
+					end
+					if control.SetDisabled then
+						local disabled = CheckOptionDisabled(v, options, path, appName)
+						control:SetDisabled(disabled)
+					end
+
+					InjectInfo(control, options, v, path, rootframe, appName)
+					container:AddChild(control)
+				end
+
+			end
+		end
+		tremove(path)
+	end
+	container:ResumeLayout()
+	container:DoLayout()
+	del(keySort)
+	del(opts)
+end
+
+local function BuildPath(path, ...)
+	for i = 1, select("#",...)  do
+		tinsert(path, (select(i,...)))
+	end
+end
+
+
+local function TreeOnButtonEnter(widget, event, uniquevalue, button)
+	local user = widget:GetUserDataTable()
+	if not user then return end
+	local options = user.options
+	local option = user.option
+	local path = user.path
+	local appName = user.appName
+
+	local feedpath = new()
+	for i = 1, #path do
+		feedpath[i] = path[i]
+	end
+
+	BuildPath(feedpath, ("\001"):split(uniquevalue))
+	local group = options
+	for i = 1, #feedpath do
+		if not group then return end
+		group = GetSubOption(group, feedpath[i])
+	end
+
+	local name = GetOptionsMemberValue("name", group, options, feedpath, appName)
+	local desc = GetOptionsMemberValue("desc", group, options, feedpath, appName)
+
+	GameTooltip:SetOwner(button, "ANCHOR_NONE")
+	if widget.type == "TabGroup" then
+		GameTooltip:SetPoint("BOTTOM",button,"TOP")
+	else
+		GameTooltip:SetPoint("LEFT",button,"RIGHT")
+	end
+
+	GameTooltip:SetText(name, 1, .82, 0, true)
+
+	if type(desc) == "string" then
+		GameTooltip:AddLine(desc, 1, 1, 1, true)
+	end
+
+	GameTooltip:Show()
+end
+
+local function TreeOnButtonLeave(widget, event, value, button)
+	GameTooltip:Hide()
+end
+
+
+local function GroupExists(appName, options, path, uniquevalue)
+	if not uniquevalue then return false end
+
+	local feedpath = new()
+	local temppath = new()
+	for i = 1, #path do
+		feedpath[i] = path[i]
+	end
+
+	BuildPath(feedpath, ("\001"):split(uniquevalue))
+
+	local group = options
+	for i = 1, #feedpath do
+		local v = feedpath[i]
+		temppath[i] = v
+		group = GetSubOption(group, v)
+
+		if not group or group.type ~= "group" or CheckOptionHidden(group, options, temppath, appName) then
+			del(feedpath)
+			del(temppath)
+			return false
+		end
+	end
+	del(feedpath)
+	del(temppath)
+	return true
+end
+
+local function GroupSelected(widget, event, uniquevalue)
+
+	local user = widget:GetUserDataTable()
+
+	local options = user.options
+	local option = user.option
+	local path = user.path
+	local rootframe = user.rootframe
+
+	local feedpath = new()
+	for i = 1, #path do
+		feedpath[i] = path[i]
+	end
+
+	BuildPath(feedpath, ("\001"):split(uniquevalue))
+	local group = options
+	for i = 1, #feedpath do
+		group = GetSubOption(group, feedpath[i])
+	end
+	widget:ReleaseChildren()
+	AceConfigDialog:FeedGroup(user.appName,options,widget,rootframe,feedpath)
+
+	del(feedpath)
+end
+
+
+
+--[[
+-- INTERNAL --
+This function will feed one group, and any inline child groups into the given container
+Select Groups will only have the selection control (tree, tabs, dropdown) fed in
+and have a group selected, this event will trigger the feeding of child groups
+
+Rules:
+	If the group is Inline, FeedOptions
+	If the group has no child groups, FeedOptions
+
+	If the group is a tab or select group, FeedOptions then add the Group Control
+	If the group is a tree group FeedOptions then
+		its parent isnt a tree group:  then add the tree control containing this and all child tree groups
+		if its parent is a tree group, its already a node on a tree
+--]]
+
+function AceConfigDialog:FeedGroup(appName,options,container,rootframe,path, isRoot)
+	local group = options
+	--follow the path to get to the curent group
+	local inline
+	local grouptype, parenttype = options.childGroups, "none"
+
+
+	for i = 1, #path do
+		local v = path[i]
+		group = GetSubOption(group, v)
+		inline = inline or pickfirstset(v.dialogInline,v.guiInline,v.inline, false)
+		parenttype = grouptype
+		grouptype = group.childGroups
+	end
+
+	if not parenttype then
+		parenttype = "tree"
+	end
+
+	--check if the group has child groups
+	local hasChildGroups
+	for k, v in pairs(group.args) do
+		if v.type == "group" and not pickfirstset(v.dialogInline,v.guiInline,v.inline, false) and not CheckOptionHidden(v, options, path, appName) then
+			hasChildGroups = true
+		end
+	end
+	if group.plugins then
+		for plugin, t in pairs(group.plugins) do
+			for k, v in pairs(t) do
+				if v.type == "group" and not pickfirstset(v.dialogInline,v.guiInline,v.inline, false) and not CheckOptionHidden(v, options, path, appName) then
+					hasChildGroups = true
+				end
+			end
+		end
+	end
+
+	container:SetLayout("flow")
+	local scroll
+
+	--Add a scrollframe if we are not going to add a group control, this is the inverse of the conditions for that later on
+	if (not (hasChildGroups and not inline)) or (grouptype ~= "tab" and grouptype ~= "select" and (parenttype == "tree" and not isRoot)) then
+		if container.type ~= "InlineGroup" and container.type ~= "SimpleGroup" then
+			scroll = gui:Create("ScrollFrame")
+			scroll:SetLayout("flow")
+			scroll.width = "fill"
+			scroll.height = "fill"
+			container:SetLayout("fill")
+			container:AddChild(scroll)
+			container = scroll
+		end
+	end
+
+	FeedOptions(appName,options,container,rootframe,path,group,nil)
+
+	if scroll then
+		container:PerformLayout()
+		local status = self:GetStatusTable(appName, path)
+		if not status.scroll then
+			status.scroll = {}
+		end
+		scroll:SetStatusTable(status.scroll)
+	end
+
+	if hasChildGroups and not inline then
+		local name = GetOptionsMemberValue("name", group, options, path, appName)
+		if grouptype == "tab" then
+
+			local tab = gui:Create("TabGroup")
+			InjectInfo(tab, options, group, path, rootframe, appName)
+			tab:SetCallback("OnGroupSelected", GroupSelected)
+			tab:SetCallback("OnTabEnter", TreeOnButtonEnter)
+			tab:SetCallback("OnTabLeave", TreeOnButtonLeave)
+
+			local status = AceConfigDialog:GetStatusTable(appName, path)
+			if not status.groups then
+				status.groups = {}
+			end
+			tab:SetStatusTable(status.groups)
+			tab.width = "fill"
+			tab.height = "fill"
+
+			local tabs = BuildGroups(group, options, path, appName)
+			tab:SetTabs(tabs)
+			tab:SetUserData("tablist", tabs)
+
+			for i = 1, #tabs do
+				local entry = tabs[i]
+				if not entry.disabled then
+					tab:SelectTab((GroupExists(appName, options, path,status.groups.selected) and status.groups.selected) or entry.value)
+					break
+				end
+			end
+
+			container:AddChild(tab)
+
+		elseif grouptype == "select" then
+
+			local select = gui:Create("DropdownGroup")
+			select:SetTitle(name)
+			InjectInfo(select, options, group, path, rootframe, appName)
+			select:SetCallback("OnGroupSelected", GroupSelected)
+			local status = AceConfigDialog:GetStatusTable(appName, path)
+			if not status.groups then
+				status.groups = {}
+			end
+			select:SetStatusTable(status.groups)
+			local grouplist, orderlist = BuildSelect(group, options, path, appName)
+			select:SetGroupList(grouplist, orderlist)
+			select:SetUserData("grouplist", grouplist)
+			select:SetUserData("orderlist", orderlist)
+
+			local firstgroup = orderlist[1]
+			if firstgroup then
+				select:SetGroup((GroupExists(appName, options, path,status.groups.selected) and status.groups.selected) or firstgroup)
+			end
+
+			select.width = "fill"
+			select.height = "fill"
+
+			container:AddChild(select)
+
+		--assume tree group by default
+		--if parenttype is tree then this group is already a node on that tree
+		elseif (parenttype ~= "tree") or isRoot then
+			local tree = gui:Create("TreeGroup")
+			InjectInfo(tree, options, group, path, rootframe, appName)
+			tree:EnableButtonTooltips(false)
+
+			tree.width = "fill"
+			tree.height = "fill"
+
+			tree:SetCallback("OnGroupSelected", GroupSelected)
+			tree:SetCallback("OnButtonEnter", TreeOnButtonEnter)
+			tree:SetCallback("OnButtonLeave", TreeOnButtonLeave)
+
+			local status = AceConfigDialog:GetStatusTable(appName, path)
+			if not status.groups then
+				status.groups = {}
+			end
+			local treedefinition = BuildGroups(group, options, path, appName, true)
+			tree:SetStatusTable(status.groups)
+
+			tree:SetTree(treedefinition)
+			tree:SetUserData("tree",treedefinition)
+
+			for i = 1, #treedefinition do
+				local entry = treedefinition[i]
+				if not entry.disabled then
+					tree:SelectByValue((GroupExists(appName, options, path,status.groups.selected) and status.groups.selected) or entry.value)
+					break
+				end
+			end
+
+			container:AddChild(tree)
+		end
+	end
+end
+
+local old_CloseSpecialWindows
+
+
+local function RefreshOnUpdate(this)
+	for appName in pairs(this.closing) do
+		if AceConfigDialog.OpenFrames[appName] then
+			AceConfigDialog.OpenFrames[appName]:Hide()
+		end
+		if AceConfigDialog.BlizOptions and AceConfigDialog.BlizOptions[appName] then
+			for key, widget in pairs(AceConfigDialog.BlizOptions[appName]) do
+				if not widget:IsVisible() then
+					widget:ReleaseChildren()
+				end
+			end
+		end
+		this.closing[appName] = nil
+	end
+
+	if this.closeAll then
+		for k, v in pairs(AceConfigDialog.OpenFrames) do
+			if not this.closeAllOverride[k] then
+				v:Hide()
+			end
+		end
+		this.closeAll = nil
+		wipe(this.closeAllOverride)
+	end
+
+	for appName in pairs(this.apps) do
+		if AceConfigDialog.OpenFrames[appName] then
+			local user = AceConfigDialog.OpenFrames[appName]:GetUserDataTable()
+			AceConfigDialog:Open(appName, unpack(user.basepath or emptyTbl))
+		end
+		if AceConfigDialog.BlizOptions and AceConfigDialog.BlizOptions[appName] then
+			for key, widget in pairs(AceConfigDialog.BlizOptions[appName]) do
+				local user = widget:GetUserDataTable()
+				if widget:IsVisible() then
+					AceConfigDialog:Open(widget:GetUserData("appName"), widget, unpack(user.basepath or emptyTbl))
+				end
+			end
+		end
+		this.apps[appName] = nil
+	end
+	this:SetScript("OnUpdate", nil)
+end
+
+-- Upgrade the OnUpdate script as well, if needed.
+if AceConfigDialog.frame:GetScript("OnUpdate") then
+	AceConfigDialog.frame:SetScript("OnUpdate", RefreshOnUpdate)
+end
+
+--- Close all open options windows
+function AceConfigDialog:CloseAll()
+	AceConfigDialog.frame.closeAll = true
+	AceConfigDialog.frame:SetScript("OnUpdate", RefreshOnUpdate)
+	if next(self.OpenFrames) then
+		return true
+	end
+end
+
+--- Close a specific options window.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+function AceConfigDialog:Close(appName)
+	if self.OpenFrames[appName] then
+		AceConfigDialog.frame.closing[appName] = true
+		AceConfigDialog.frame:SetScript("OnUpdate", RefreshOnUpdate)
+		return true
+	end
+end
+
+-- Internal -- Called by AceConfigRegistry
+function AceConfigDialog:ConfigTableChanged(event, appName)
+	AceConfigDialog.frame.apps[appName] = true
+	AceConfigDialog.frame:SetScript("OnUpdate", RefreshOnUpdate)
+end
+
+reg.RegisterCallback(AceConfigDialog, "ConfigTableChange", "ConfigTableChanged")
+
+--- Sets the default size of the options window for a specific application.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param width The default width
+-- @param height The default height
+function AceConfigDialog:SetDefaultSize(appName, width, height)
+	local status = AceConfigDialog:GetStatusTable(appName)
+	if type(width) == "number" and type(height) == "number" then
+		status.width = width
+		status.height = height
+	end
+end
+
+--- Open an option window at the specified path (if any).
+-- This function can optionally feed the group into a pre-created container
+-- instead of creating a new container frame.
+-- @paramsig appName [, container][, ...]
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param container An optional container frame to feed the options into
+-- @param ... The path to open after creating the options window (see `:SelectGroup` for details)
+function AceConfigDialog:Open(appName, container, ...)
+	if not old_CloseSpecialWindows then
+		old_CloseSpecialWindows = CloseSpecialWindows
+		CloseSpecialWindows = function()
+			local found = old_CloseSpecialWindows()
+			return self:CloseAll() or found
+		end
+	end
+	local app = reg:GetOptionsTable(appName)
+	if not app then
+		error(("%s isn't registed with AceConfigRegistry, unable to open config"):format(appName), 2)
+	end
+	local options = app("dialog", MAJOR)
+
+	local f
+
+	local path = new()
+	local name = GetOptionsMemberValue("name", options, options, path, appName)
+
+	--If an optional path is specified add it to the path table before feeding the options
+	--as container is optional as well it may contain the first element of the path
+	if type(container) == "string" then
+		tinsert(path, container)
+		container = nil
+	end
+	for n = 1, select("#",...) do
+		tinsert(path, (select(n, ...)))
+	end
+
+	local option = options
+	if type(container) == "table" and container.type == "BlizOptionsGroup" and #path > 0 then
+		for i = 1, #path do
+			option = options.args[path[i]]
+		end
+		name = format("%s - %s", name, GetOptionsMemberValue("name", option, options, path, appName))
+	end
+
+	--if a container is given feed into that
+	if container then
+		f = container
+		f:ReleaseChildren()
+		f:SetUserData("appName", appName)
+		f:SetUserData("iscustom", true)
+		if #path > 0 then
+			f:SetUserData("basepath", copy(path))
+		end
+		local status = AceConfigDialog:GetStatusTable(appName)
+		if not status.width then
+			status.width =  700
+		end
+		if not status.height then
+			status.height = 500
+		end
+		if f.SetStatusTable then
+			f:SetStatusTable(status)
+		end
+		if f.SetTitle then
+			f:SetTitle(name or "")
+		end
+	else
+		if not self.OpenFrames[appName] then
+			f = gui:Create("Frame")
+			self.OpenFrames[appName] = f
+		else
+			f = self.OpenFrames[appName]
+		end
+		f:ReleaseChildren()
+		f:SetCallback("OnClose", FrameOnClose)
+		f:SetUserData("appName", appName)
+		if #path > 0 then
+			f:SetUserData("basepath", copy(path))
+		end
+		f:SetTitle(name or "")
+		local status = AceConfigDialog:GetStatusTable(appName)
+		f:SetStatusTable(status)
+	end
+
+	self:FeedGroup(appName,options,f,f,path,true)
+	if f.Show then
+		f:Show()
+	end
+	del(path)
+
+	if AceConfigDialog.frame.closeAll then
+		-- close all is set, but thats not good, since we're just opening here, so force it
+		AceConfigDialog.frame.closeAllOverride[appName] = true
+	end
+end
+
+-- convert pre-39 BlizOptions structure to the new format
+if oldminor and oldminor < 39 and AceConfigDialog.BlizOptions then
+	local old = AceConfigDialog.BlizOptions
+	local new = {}
+	for key, widget in pairs(old) do
+		local appName = widget:GetUserData("appName")
+		if not new[appName] then new[appName] = {} end
+		new[appName][key] = widget
+	end
+	AceConfigDialog.BlizOptions = new
+else
+	AceConfigDialog.BlizOptions = AceConfigDialog.BlizOptions or {}
+end
+
+local function FeedToBlizPanel(widget, event)
+	local path = widget:GetUserData("path")
+	AceConfigDialog:Open(widget:GetUserData("appName"), widget, unpack(path or emptyTbl))
+end
+
+local function ClearBlizPanel(widget, event)
+	local appName = widget:GetUserData("appName")
+	AceConfigDialog.frame.closing[appName] = true
+	AceConfigDialog.frame:SetScript("OnUpdate", RefreshOnUpdate)
+end
+
+--- Add an option table into the Blizzard Interface Options panel.
+-- You can optionally supply a descriptive name to use and a parent frame to use,
+-- as well as a path in the options table.\\
+-- If no name is specified, the appName will be used instead.
+--
+-- If you specify a proper `parent` (by name), the interface options will generate a
+-- tree layout. Note that only one level of children is supported, so the parent always
+-- has to be a head-level note.
+--
+-- This function returns a reference to the container frame registered with the Interface
+-- Options. You can use this reference to open the options with the API function
+-- `InterfaceOptionsFrame_OpenToCategory`.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param name A descriptive name to display in the options tree (defaults to appName)
+-- @param parent The parent to use in the interface options tree.
+-- @param ... The path in the options table to feed into the interface options panel.
+-- @return The reference to the frame registered into the Interface Options.
+function AceConfigDialog:AddToBlizOptions(appName, name, parent, ...)
+	local BlizOptions = AceConfigDialog.BlizOptions
+
+	local key = appName
+	for n = 1, select("#", ...) do
+		key = key.."\001"..select(n, ...)
+	end
+
+	if not BlizOptions[appName] then
+		BlizOptions[appName] = {}
+	end
+
+	if not BlizOptions[appName][key] then
+		local group = gui:Create("BlizOptionsGroup")
+		BlizOptions[appName][key] = group
+		group:SetName(name or appName, parent)
+
+		group:SetTitle(name or appName)
+		group:SetUserData("appName", appName)
+		if select("#", ...) > 0 then
+			local path = {}
+			for n = 1, select("#",...) do
+				tinsert(path, (select(n, ...)))
+			end
+			group:SetUserData("path", path)
+		end
+		group:SetCallback("OnShow", FeedToBlizPanel)
+		group:SetCallback("OnHide", ClearBlizPanel)
+		InterfaceOptions_AddCategory(group.frame)
+		return group.frame
+	else
+		error(("%s has already been added to the Blizzard Options Window with the given path"):format(appName), 2)
+	end
+end
diff --git a/Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.xml b/Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.xml
new file mode 100755
index 0000000..068be6f
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.xml
@@ -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="AceConfigDialog-3.0.lua"/>
+</Ui>
diff --git a/Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.lua b/Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.lua
new file mode 100755
index 0000000..e7a6d76
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.lua
@@ -0,0 +1,350 @@
+--- AceConfigRegistry-3.0 handles central registration of options tables in use by addons and modules.\\
+-- Options tables can be registered as raw tables, OR as function refs that return a table.\\
+-- Such functions receive three arguments: "uiType", "uiName", "appName". \\
+-- * Valid **uiTypes**: "cmd", "dropdown", "dialog". This is verified by the library at call time. \\
+-- * The **uiName** field is expected to contain the full name of the calling addon, including version, e.g. "FooBar-1.0". This is verified by the library at call time.\\
+-- * The **appName** field is the options table name as given at registration time \\
+--
+-- :IterateOptionsTables() (and :GetOptionsTable() if only given one argument) return a function reference that the requesting config handling addon must call with valid "uiType", "uiName".
+-- @class file
+-- @name AceConfigRegistry-3.0
+-- @release $Id: AceConfigRegistry-3.0.lua 1169 2018-02-27 16:18:28Z nevcairiel $
+local CallbackHandler = LibStub("CallbackHandler-1.0")
+
+local MAJOR, MINOR = "AceConfigRegistry-3.0", 18
+local AceConfigRegistry = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not AceConfigRegistry then return end
+
+AceConfigRegistry.tables = AceConfigRegistry.tables or {}
+
+if not AceConfigRegistry.callbacks then
+	AceConfigRegistry.callbacks = CallbackHandler:New(AceConfigRegistry)
+end
+
+-- Lua APIs
+local tinsert, tconcat = table.insert, table.concat
+local strfind, strmatch = string.find, string.match
+local type, tostring, select, pairs = type, tostring, select, pairs
+local error, assert = error, assert
+
+-----------------------------------------------------------------------
+-- Validating options table consistency:
+
+
+AceConfigRegistry.validated = {
+	-- list of options table names ran through :ValidateOptionsTable automatically.
+	-- CLEARED ON PURPOSE, since newer versions may have newer validators
+	cmd = {},
+	dropdown = {},
+	dialog = {},
+}
+
+
+
+local function err(msg, errlvl, ...)
+	local t = {}
+	for i=select("#",...),1,-1 do
+		tinsert(t, (select(i, ...)))
+	end
+	error(MAJOR..":ValidateOptionsTable(): "..tconcat(t,".")..msg, errlvl+2)
+end
+
+
+local isstring={["string"]=true, _="string"}
+local isstringfunc={["string"]=true,["function"]=true, _="string or funcref"}
+local istable={["table"]=true,   _="table"}
+local ismethodtable={["table"]=true,["string"]=true,["function"]=true,   _="methodname, funcref or table"}
+local optstring={["nil"]=true,["string"]=true, _="string"}
+local optstringfunc={["nil"]=true,["string"]=true,["function"]=true, _="string or funcref"}
+local optstringnumberfunc={["nil"]=true,["string"]=true,["number"]=true,["function"]=true, _="string, number or funcref"}
+local optnumber={["nil"]=true,["number"]=true, _="number"}
+local optmethod={["nil"]=true,["string"]=true,["function"]=true, _="methodname or funcref"}
+local optmethodfalse={["nil"]=true,["string"]=true,["function"]=true,["boolean"]={[false]=true},  _="methodname, funcref or false"}
+local optmethodnumber={["nil"]=true,["string"]=true,["function"]=true,["number"]=true,  _="methodname, funcref or number"}
+local optmethodtable={["nil"]=true,["string"]=true,["function"]=true,["table"]=true,  _="methodname, funcref or table"}
+local optmethodbool={["nil"]=true,["string"]=true,["function"]=true,["boolean"]=true,  _="methodname, funcref or boolean"}
+local opttable={["nil"]=true,["table"]=true,  _="table"}
+local optbool={["nil"]=true,["boolean"]=true,  _="boolean"}
+local optboolnumber={["nil"]=true,["boolean"]=true,["number"]=true,  _="boolean or number"}
+local optstringnumber={["nil"]=true,["string"]=true,["number"]=true, _="string or number"}
+
+local basekeys={
+	type=isstring,
+	name=isstringfunc,
+	desc=optstringfunc,
+	descStyle=optstring,
+	order=optmethodnumber,
+	validate=optmethodfalse,
+	confirm=optmethodbool,
+	confirmText=optstring,
+	disabled=optmethodbool,
+	hidden=optmethodbool,
+		guiHidden=optmethodbool,
+		dialogHidden=optmethodbool,
+		dropdownHidden=optmethodbool,
+	cmdHidden=optmethodbool,
+	icon=optstringnumberfunc,
+	iconCoords=optmethodtable,
+	handler=opttable,
+	get=optmethodfalse,
+	set=optmethodfalse,
+	func=optmethodfalse,
+	arg={["*"]=true},
+	width=optstringnumber,
+}
+
+local typedkeys={
+	header={},
+	description={
+		image=optstringnumberfunc,
+		imageCoords=optmethodtable,
+		imageHeight=optnumber,
+		imageWidth=optnumber,
+		fontSize=optstringfunc,
+	},
+	group={
+		args=istable,
+		plugins=opttable,
+		inline=optbool,
+			cmdInline=optbool,
+			guiInline=optbool,
+			dropdownInline=optbool,
+			dialogInline=optbool,
+		childGroups=optstring,
+	},
+	execute={
+		image=optstringnumberfunc,
+		imageCoords=optmethodtable,
+		imageHeight=optnumber,
+		imageWidth=optnumber,
+	},
+	input={
+		pattern=optstring,
+		usage=optstring,
+		control=optstring,
+		dialogControl=optstring,
+		dropdownControl=optstring,
+		multiline=optboolnumber,
+	},
+	toggle={
+		tristate=optbool,
+		image=optstringnumberfunc,
+		imageCoords=optmethodtable,
+	},
+	tristate={
+	},
+	range={
+		min=optnumber,
+		softMin=optnumber,
+		max=optnumber,
+		softMax=optnumber,
+		step=optnumber,
+		bigStep=optnumber,
+		isPercent=optbool,
+	},
+	select={
+		values=ismethodtable,
+		style={
+			["nil"]=true,
+			["string"]={dropdown=true,radio=true},
+			_="string: 'dropdown' or 'radio'"
+		},
+		control=optstring,
+		dialogControl=optstring,
+		dropdownControl=optstring,
+		itemControl=optstring,
+	},
+	multiselect={
+		values=ismethodtable,
+		style=optstring,
+		tristate=optbool,
+		control=optstring,
+		dialogControl=optstring,
+		dropdownControl=optstring,
+	},
+	color={
+		hasAlpha=optmethodbool,
+	},
+	keybinding={
+		-- TODO
+	},
+}
+
+local function validateKey(k,errlvl,...)
+	errlvl=(errlvl or 0)+1
+	if type(k)~="string" then
+		err("["..tostring(k).."] - key is not a string", errlvl,...)
+	end
+	if strfind(k, "[%c\127]") then
+		err("["..tostring(k).."] - key name contained control characters", errlvl,...)
+	end
+end
+
+local function validateVal(v, oktypes, errlvl,...)
+	errlvl=(errlvl or 0)+1
+	local isok=oktypes[type(v)] or oktypes["*"]
+
+	if not isok then
+		err(": expected a "..oktypes._..", got '"..tostring(v).."'", errlvl,...)
+	end
+	if type(isok)=="table" then		-- isok was a table containing specific values to be tested for!
+		if not isok[v] then
+			err(": did not expect "..type(v).." value '"..tostring(v).."'", errlvl,...)
+		end
+	end
+end
+
+local function validate(options,errlvl,...)
+	errlvl=(errlvl or 0)+1
+	-- basic consistency
+	if type(options)~="table" then
+		err(": expected a table, got a "..type(options), errlvl,...)
+	end
+	if type(options.type)~="string" then
+		err(".type: expected a string, got a "..type(options.type), errlvl,...)
+	end
+
+	-- get type and 'typedkeys' member
+	local tk = typedkeys[options.type]
+	if not tk then
+		err(".type: unknown type '"..options.type.."'", errlvl,...)
+	end
+
+	-- make sure that all options[] are known parameters
+	for k,v in pairs(options) do
+		if not (tk[k] or basekeys[k]) then
+			err(": unknown parameter", errlvl,tostring(k),...)
+		end
+	end
+
+	-- verify that required params are there, and that everything is the right type
+	for k,oktypes in pairs(basekeys) do
+		validateVal(options[k], oktypes, errlvl,k,...)
+	end
+	for k,oktypes in pairs(tk) do
+		validateVal(options[k], oktypes, errlvl,k,...)
+	end
+
+	-- extra logic for groups
+	if options.type=="group" then
+		for k,v in pairs(options.args) do
+			validateKey(k,errlvl,"args",...)
+			validate(v, errlvl,k,"args",...)
+		end
+		if options.plugins then
+			for plugname,plugin in pairs(options.plugins) do
+				if type(plugin)~="table" then
+					err(": expected a table, got '"..tostring(plugin).."'", errlvl,tostring(plugname),"plugins",...)
+				end
+				for k,v in pairs(plugin) do
+					validateKey(k,errlvl,tostring(plugname),"plugins",...)
+					validate(v, errlvl,k,tostring(plugname),"plugins",...)
+				end
+			end
+		end
+	end
+end
+
+
+--- Validates basic structure and integrity of an options table \\
+-- Does NOT verify that get/set etc actually exist, since they can be defined at any depth
+-- @param options The table to be validated
+-- @param name The name of the table to be validated (shown in any error message)
+-- @param errlvl (optional number) error level offset, default 0 (=errors point to the function calling :ValidateOptionsTable)
+function AceConfigRegistry:ValidateOptionsTable(options,name,errlvl)
+	errlvl=(errlvl or 0)+1
+	name = name or "Optionstable"
+	if not options.name then
+		options.name=name	-- bit of a hack, the root level doesn't really need a .name :-/
+	end
+	validate(options,errlvl,name)
+end
+
+--- Fires a "ConfigTableChange" callback for those listening in on it, allowing config GUIs to refresh.
+-- You should call this function if your options table changed from any outside event, like a game event
+-- or a timer.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+function AceConfigRegistry:NotifyChange(appName)
+	if not AceConfigRegistry.tables[appName] then return end
+	AceConfigRegistry.callbacks:Fire("ConfigTableChange", appName)
+end
+
+-- -------------------------------------------------------------------
+-- Registering and retreiving options tables:
+
+
+-- validateGetterArgs: helper function for :GetOptionsTable (or, rather, the getter functions returned by it)
+
+local function validateGetterArgs(uiType, uiName, errlvl)
+	errlvl=(errlvl or 0)+2
+	if uiType~="cmd" and uiType~="dropdown" and uiType~="dialog" then
+		error(MAJOR..": Requesting options table: 'uiType' - invalid configuration UI type, expected 'cmd', 'dropdown' or 'dialog'", errlvl)
+	end
+	if not strmatch(uiName, "[A-Za-z]%-[0-9]") then	-- Expecting e.g. "MyLib-1.2"
+		error(MAJOR..": Requesting options table: 'uiName' - badly formatted or missing version number. Expected e.g. 'MyLib-1.2'", errlvl)
+	end
+end
+
+--- Register an options table with the config registry.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param options The options table, OR a function reference that generates it on demand. \\
+-- See the top of the page for info on arguments passed to such functions.
+-- @param skipValidation Skip options table validation (primarily useful for extremely huge options, with a noticeable slowdown)
+function AceConfigRegistry:RegisterOptionsTable(appName, options, skipValidation)
+	if type(options)=="table" then
+		if options.type~="group" then	-- quick sanity checker
+			error(MAJOR..": RegisterOptionsTable(appName, options): 'options' - missing type='group' member in root group", 2)
+		end
+		AceConfigRegistry.tables[appName] = function(uiType, uiName, errlvl)
+			errlvl=(errlvl or 0)+1
+			validateGetterArgs(uiType, uiName, errlvl)
+			if not AceConfigRegistry.validated[uiType][appName] and not skipValidation then
+				AceConfigRegistry:ValidateOptionsTable(options, appName, errlvl)	-- upgradable
+				AceConfigRegistry.validated[uiType][appName] = true
+			end
+			return options
+		end
+	elseif type(options)=="function" then
+		AceConfigRegistry.tables[appName] = function(uiType, uiName, errlvl)
+			errlvl=(errlvl or 0)+1
+			validateGetterArgs(uiType, uiName, errlvl)
+			local tab = assert(options(uiType, uiName, appName))
+			if not AceConfigRegistry.validated[uiType][appName] and not skipValidation then
+				AceConfigRegistry:ValidateOptionsTable(tab, appName, errlvl)	-- upgradable
+				AceConfigRegistry.validated[uiType][appName] = true
+			end
+			return tab
+		end
+	else
+		error(MAJOR..": RegisterOptionsTable(appName, options): 'options' - expected table or function reference", 2)
+	end
+end
+
+--- Returns an iterator of ["appName"]=funcref pairs
+function AceConfigRegistry:IterateOptionsTables()
+	return pairs(AceConfigRegistry.tables)
+end
+
+
+
+
+--- Query the registry for a specific options table.
+-- If only appName is given, a function is returned which you
+-- can call with (uiType,uiName) to get the table.\\
+-- If uiType&uiName are given, the table is returned.
+-- @param appName The application name as given to `:RegisterOptionsTable()`
+-- @param uiType The type of UI to get the table for, one of "cmd", "dropdown", "dialog"
+-- @param uiName The name of the library/addon querying for the table, e.g. "MyLib-1.0"
+function AceConfigRegistry:GetOptionsTable(appName, uiType, uiName)
+	local f = AceConfigRegistry.tables[appName]
+	if not f then
+		return nil
+	end
+
+	if uiType then
+		return f(uiType,uiName,1)	-- get the table for us
+	else
+		return f	-- return the function
+	end
+end
diff --git a/Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.xml b/Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.xml
new file mode 100755
index 0000000..5989072
--- /dev/null
+++ b/Titan/libs/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.xml
@@ -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="AceConfigRegistry-3.0.lua"/>
+</Ui>
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets.toc b/Titan/libs/AceGUI-3.0-SharedMediaWidgets.toc
new file mode 100644
index 0000000..3c55ea5
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets.toc
@@ -0,0 +1,20 @@
+## Interface: 70000
+## X-Curse-Packaged-Version: r57-release
+## X-Curse-Project-Name: AceGUI-3.0-SharedMediaWidgets
+## X-Curse-Project-ID: ace-gui-3-0-shared-media-widgets
+## X-Curse-Repository-ID: wow/ace-gui-3-0-shared-media-widgets/mainline
+
+## Title: Lib: AceGUI-3.0-SharedMediaWidgets
+## Notes: Enables AceGUI-3.0 widgets for the 5 basic SharedMedia-3.0 types
+## Author: Yssaril
+## OptionalDeps: Ace3, LibSharedMedia-3.0
+## X-Category: Library
+
+#@no-lib-strip@
+Libs\Libstub\Libstub.lua
+Libs\CallbackHandler-1.0\CallbackHandler-1.0.xml
+Libs\AceGUI-3.0\AceGUI-3.0.xml
+Libs\LibSharedMedia-3.0\lib.xml
+#@end-no-lib-strip@
+
+widget.xml
\ No newline at end of file
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets/BackgroundWidget.lua b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/BackgroundWidget.lua
new file mode 100644
index 0000000..1601747
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/BackgroundWidget.lua
@@ -0,0 +1,235 @@
+-- Widget is based on the AceGUIWidget-DropDown.lua supplied with AceGUI-3.0
+-- Widget created by Yssaril
+
+local AceGUI = LibStub("AceGUI-3.0")
+local Media = LibStub("LibSharedMedia-3.0")
+
+local AGSMW = LibStub("AceGUISharedMediaWidgets-1.0")
+
+do
+	local widgetType = "LSM30_Background"
+	local widgetVersion = 11
+
+	local contentFrameCache = {}
+	local function ReturnSelf(self)
+		self:ClearAllPoints()
+		self:Hide()
+		self.check:Hide()
+		table.insert(contentFrameCache, self)
+	end
+
+	local function ContentOnClick(this, button)
+		local self = this.obj
+		self:Fire("OnValueChanged", this.text:GetText())
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function ContentOnEnter(this, button)
+		local self = this.obj
+		local text = this.text:GetText()
+		local background = self.list[text] ~= text and self.list[text] or Media:Fetch('background',text)
+		self.dropdown.bgTex:SetTexture(background)
+	end
+
+	local function GetContentLine()
+		local frame
+		if next(contentFrameCache) then
+			frame = table.remove(contentFrameCache)
+		else
+			frame = CreateFrame("Button", nil, UIParent)
+				--frame:SetWidth(200)
+				frame:SetHeight(18)
+				frame:SetHighlightTexture([[Interface\QuestFrame\UI-QuestTitleHighlight]], "ADD")
+				frame:SetScript("OnClick", ContentOnClick)
+				frame:SetScript("OnEnter", ContentOnEnter)
+
+			local check = frame:CreateTexture("OVERLAY")
+				check:SetWidth(16)
+				check:SetHeight(16)
+				check:SetPoint("LEFT",frame,"LEFT",1,-1)
+				check:SetTexture("Interface\\Buttons\\UI-CheckBox-Check")
+				check:Hide()
+			frame.check = check
+
+			local text = frame:CreateFontString(nil,"OVERLAY","GameFontWhite")
+				local font, size = text:GetFont()
+				text:SetFont(font,size,"OUTLINE")
+
+				text:SetPoint("TOPLEFT", check, "TOPRIGHT", 1, 0)
+				text:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT", -2, 0)
+				text:SetJustifyH("LEFT")
+				text:SetText("Test Test Test Test Test Test Test")
+			frame.text = text
+
+			frame.ReturnSelf = ReturnSelf
+		end
+		frame:Show()
+		return frame
+	end
+
+	local function OnAcquire(self)
+		self:SetHeight(44)
+		self:SetWidth(200)
+	end
+
+	local function OnRelease(self)
+		self:SetText("")
+		self:SetLabel("")
+		self:SetDisabled(false)
+
+		self.value = nil
+		self.list = nil
+		self.open = nil
+		self.hasClose = nil
+
+		self.frame:ClearAllPoints()
+		self.frame:Hide()
+	end
+
+	local function SetValue(self, value) -- Set the value to an item in the List.
+		if self.list then
+			self:SetText(value or "")
+		end
+		self.value = value
+	end
+
+	local function GetValue(self)
+		return self.value
+	end
+
+	local function SetList(self, list) -- Set the list of values for the dropdown (key => value pairs)
+		self.list = list or Media:HashTable("background")
+	end
+
+
+	local function SetText(self, text) -- Set the text displayed in the box.
+		self.frame.text:SetText(text or "")
+		local background = self.list[text] ~= text and self.list[text] or Media:Fetch('background',text)
+
+		self.frame.displayButton:SetBackdrop({bgFile = background,
+			edgeFile = "Interface/Tooltips/UI-Tooltip-Border",
+			edgeSize = 16,
+			insets = { left = 4, right = 4, top = 4, bottom = 4 }})
+	end
+
+	local function SetLabel(self, text) -- Set the text for the label.
+		self.frame.label:SetText(text or "")
+	end
+
+	local function AddItem(self, key, value) -- Add an item to the list.
+		self.list = self.list or {}
+		self.list[key] = value
+	end
+	local SetItemValue = AddItem -- Set the value of a item in the list. <<same as adding a new item>>
+
+	local function SetMultiselect(self, flag) end -- Toggle multi-selecting. <<Dummy function to stay inline with the dropdown API>>
+	local function GetMultiselect() return false end-- Query the multi-select flag. <<Dummy function to stay inline with the dropdown API>>
+	local function SetItemDisabled(self, key) end-- Disable one item in the list. <<Dummy function to stay inline with the dropdown API>>
+
+	local function SetDisabled(self, disabled) -- Disable the widget.
+		self.disabled = disabled
+		if disabled then
+			self.frame:Disable()
+			self.frame.displayButton:SetBackdropColor(.2,.2,.2,1)
+		else
+			self.frame:Enable()
+			self.frame.displayButton:SetBackdropColor(1,1,1,1)
+		end
+	end
+
+	local function textSort(a,b)
+		return string.upper(a) < string.upper(b)
+	end
+
+	local sortedlist = {}
+	local function ToggleDrop(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+			AceGUI:ClearFocus()
+		else
+			AceGUI:SetFocus(self)
+			self.dropdown = AGSMW:GetDropDownFrame()
+			local width = self.frame:GetWidth()
+			self.dropdown:SetPoint("TOPLEFT", self.frame, "BOTTOMLEFT")
+			self.dropdown:SetPoint("TOPRIGHT", self.frame, "BOTTOMRIGHT", width < 160 and (160 - width) or 0, 0)
+			for k, v in pairs(self.list) do
+				sortedlist[#sortedlist+1] = k
+			end
+			table.sort(sortedlist, textSort)
+			for i, k in ipairs(sortedlist) do
+				local f = GetContentLine()
+				f.text:SetText(k)
+				--print(k)
+				if k == self.value then
+					f.check:Show()
+				end
+				f.obj = self
+				f.dropdown = self.dropdown
+				self.dropdown:AddFrame(f)
+			end
+			wipe(sortedlist)
+		end
+	end
+
+	local function ClearFocus(self)
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function OnHide(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function Drop_OnEnter(this)
+		this.obj:Fire("OnEnter")
+	end
+
+	local function Drop_OnLeave(this)
+		this.obj:Fire("OnLeave")
+	end
+
+	local function Constructor()
+		local frame = AGSMW:GetBaseFrameWithWindow()
+		local self = {}
+
+		self.type = widgetType
+		self.frame = frame
+		frame.obj = self
+		frame.dropButton.obj = self
+		frame.dropButton:SetScript("OnEnter", Drop_OnEnter)
+		frame.dropButton:SetScript("OnLeave", Drop_OnLeave)
+		frame.dropButton:SetScript("OnClick",ToggleDrop)
+		frame:SetScript("OnHide", OnHide)
+
+		self.alignoffset = 31
+
+		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.ToggleDrop = ToggleDrop
+
+		AceGUI:RegisterAsWidget(self)
+		return self
+	end
+
+	AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion)
+
+end
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets/BorderWidget.lua b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/BorderWidget.lua
new file mode 100644
index 0000000..b4f31c6
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/BorderWidget.lua
@@ -0,0 +1,230 @@
+-- Widget is based on the AceGUIWidget-DropDown.lua supplied with AceGUI-3.0
+-- Widget created by Yssaril
+
+local AceGUI = LibStub("AceGUI-3.0")
+local Media = LibStub("LibSharedMedia-3.0")
+
+local AGSMW = LibStub("AceGUISharedMediaWidgets-1.0")
+
+do
+	local widgetType = "LSM30_Border"
+	local widgetVersion = 11
+
+	local contentFrameCache = {}
+	local function ReturnSelf(self)
+		self:ClearAllPoints()
+		self:Hide()
+		self.check:Hide()
+		table.insert(contentFrameCache, self)
+	end
+
+	local function ContentOnClick(this, button)
+		local self = this.obj
+		self:Fire("OnValueChanged", this.text:GetText())
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function ContentOnEnter(this, button)
+		local self = this.obj
+		local text = this.text:GetText()
+		local border = self.list[text] ~= text and self.list[text] or Media:Fetch('border',text)
+		this.dropdown:SetBackdrop({edgeFile = border,
+			bgFile=[[Interface\DialogFrame\UI-DialogBox-Background-Dark]],
+			tile = true, tileSize = 16, edgeSize = 16,
+			insets = { left = 4, right = 4, top = 4, bottom = 4 }})
+	end
+
+	local function GetContentLine()
+		local frame
+		if next(contentFrameCache) then
+			frame = table.remove(contentFrameCache)
+		else
+			frame = CreateFrame("Button", nil, UIParent)
+				--frame:SetWidth(200)
+				frame:SetHeight(18)
+				frame:SetHighlightTexture([[Interface\QuestFrame\UI-QuestTitleHighlight]], "ADD")
+				frame:SetScript("OnClick", ContentOnClick)
+				frame:SetScript("OnEnter", ContentOnEnter)
+			local check = frame:CreateTexture("OVERLAY")
+				check:SetWidth(16)
+				check:SetHeight(16)
+				check:SetPoint("LEFT",frame,"LEFT",1,-1)
+				check:SetTexture("Interface\\Buttons\\UI-CheckBox-Check")
+				check:Hide()
+			frame.check = check
+			local text = frame:CreateFontString(nil,"OVERLAY","GameFontWhite")
+				text:SetPoint("TOPLEFT", check, "TOPRIGHT", 1, 0)
+				text:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT", -2, 0)
+				text:SetJustifyH("LEFT")
+				text:SetText("Test Test Test Test Test Test Test")
+			frame.text = text
+			frame.ReturnSelf = ReturnSelf
+		end
+		frame:Show()
+		return frame
+	end
+
+	local function OnAcquire(self)
+		self:SetHeight(44)
+		self:SetWidth(200)
+	end
+
+	local function OnRelease(self)
+		self:SetText("")
+		self:SetLabel("")
+		self:SetDisabled(false)
+
+		self.value = nil
+		self.list = nil
+		self.open = nil
+		self.hasClose = nil
+
+		self.frame:ClearAllPoints()
+		self.frame:Hide()
+	end
+
+	local function SetValue(self, value) -- Set the value to an item in the List.
+		if self.list then
+			self:SetText(value or "")
+		end
+		self.value = value
+	end
+
+	local function GetValue(self)
+		return self.value
+	end
+
+	local function SetList(self, list) -- Set the list of values for the dropdown (key => value pairs)
+		self.list = list or Media:HashTable("border")
+	end
+
+
+	local function SetText(self, text) -- Set the text displayed in the box.
+		self.frame.text:SetText(text or "")
+		local border = self.list[text] ~= text and self.list[text] or Media:Fetch('border',text)
+
+		self.frame.displayButton:SetBackdrop({edgeFile = border,
+			bgFile=[[Interface\DialogFrame\UI-DialogBox-Background-Dark]],
+			tile = true, tileSize = 16, edgeSize = 16,
+			insets = { left = 4, right = 4, top = 4, bottom = 4 }})
+	end
+
+	local function SetLabel(self, text) -- Set the text for the label.
+		self.frame.label:SetText(text or "")
+	end
+
+	local function AddItem(self, key, value) -- Add an item to the list.
+		self.list = self.list or {}
+		self.list[key] = value
+	end
+	local SetItemValue = AddItem -- Set the value of a item in the list. <<same as adding a new item>>
+
+	local function SetMultiselect(self, flag) end -- Toggle multi-selecting. <<Dummy function to stay inline with the dropdown API>>
+	local function GetMultiselect() return false end-- Query the multi-select flag. <<Dummy function to stay inline with the dropdown API>>
+	local function SetItemDisabled(self, key) end-- Disable one item in the list. <<Dummy function to stay inline with the dropdown API>>
+
+	local function SetDisabled(self, disabled) -- Disable the widget.
+		self.disabled = disabled
+		if disabled then
+			self.frame:Disable()
+		else
+			self.frame:Enable()
+		end
+	end
+
+	local function textSort(a,b)
+		return string.upper(a) < string.upper(b)
+	end
+
+	local sortedlist = {}
+	local function ToggleDrop(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+			AceGUI:ClearFocus()
+		else
+			AceGUI:SetFocus(self)
+			self.dropdown = AGSMW:GetDropDownFrame()
+			local width = self.frame:GetWidth()
+			self.dropdown:SetPoint("TOPLEFT", self.frame, "BOTTOMLEFT")
+			self.dropdown:SetPoint("TOPRIGHT", self.frame, "BOTTOMRIGHT", width < 160 and (160 - width) or 0, 0)
+			for k, v in pairs(self.list) do
+				sortedlist[#sortedlist+1] = k
+			end
+			table.sort(sortedlist, textSort)
+			for i, k in ipairs(sortedlist) do
+				local f = GetContentLine()
+				f.text:SetText(k)
+				--print(k)
+				if k == self.value then
+					f.check:Show()
+				end
+				f.obj = self
+				f.dropdown = self.dropdown
+				self.dropdown:AddFrame(f)
+			end
+			wipe(sortedlist)
+		end
+	end
+
+	local function ClearFocus(self)
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function OnHide(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function Drop_OnEnter(this)
+		this.obj:Fire("OnEnter")
+	end
+
+	local function Drop_OnLeave(this)
+		this.obj:Fire("OnLeave")
+	end
+
+	local function Constructor()
+		local frame = AGSMW:GetBaseFrameWithWindow()
+		local self = {}
+
+		self.type = widgetType
+		self.frame = frame
+		frame.obj = self
+		frame.dropButton.obj = self
+		frame.dropButton:SetScript("OnEnter", Drop_OnEnter)
+		frame.dropButton:SetScript("OnLeave", Drop_OnLeave)
+		frame.dropButton:SetScript("OnClick",ToggleDrop)
+		frame:SetScript("OnHide", OnHide)
+
+		self.alignoffset = 31
+
+		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.ToggleDrop = ToggleDrop
+
+		AceGUI:RegisterAsWidget(self)
+		return self
+	end
+
+	AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion)
+
+end
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets/FontWidget.lua b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/FontWidget.lua
new file mode 100644
index 0000000..cd2b5db
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/FontWidget.lua
@@ -0,0 +1,216 @@
+-- Widget is based on the AceGUIWidget-DropDown.lua supplied with AceGUI-3.0
+-- Widget created by Yssaril
+
+local AceGUI = LibStub("AceGUI-3.0")
+local Media = LibStub("LibSharedMedia-3.0")
+
+local AGSMW = LibStub("AceGUISharedMediaWidgets-1.0")
+
+do
+	local widgetType = "LSM30_Font"
+	local widgetVersion = 11
+
+	local contentFrameCache = {}
+	local function ReturnSelf(self)
+		self:ClearAllPoints()
+		self:Hide()
+		self.check:Hide()
+		table.insert(contentFrameCache, self)
+	end
+
+	local function ContentOnClick(this, button)
+		local self = this.obj
+		self:Fire("OnValueChanged", this.text:GetText())
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function GetContentLine()
+		local frame
+		if next(contentFrameCache) then
+			frame = table.remove(contentFrameCache)
+		else
+			frame = CreateFrame("Button", nil, UIParent)
+				--frame:SetWidth(200)
+				frame:SetHeight(18)
+				frame:SetHighlightTexture([[Interface\QuestFrame\UI-QuestTitleHighlight]], "ADD")
+				frame:SetScript("OnClick", ContentOnClick)
+			local check = frame:CreateTexture("OVERLAY")
+				check:SetWidth(16)
+				check:SetHeight(16)
+				check:SetPoint("LEFT",frame,"LEFT",1,-1)
+				check:SetTexture("Interface\\Buttons\\UI-CheckBox-Check")
+				check:Hide()
+			frame.check = check
+			local text = frame:CreateFontString(nil,"OVERLAY","GameFontWhite")
+				text:SetPoint("TOPLEFT", check, "TOPRIGHT", 1, 0)
+				text:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT", -2, 0)
+				text:SetJustifyH("LEFT")
+				text:SetText("Test Test Test Test Test Test Test")
+			frame.text = text
+			frame.ReturnSelf = ReturnSelf
+		end
+		frame:Show()
+		return frame
+	end
+
+	local function OnAcquire(self)
+		self:SetHeight(44)
+		self:SetWidth(200)
+	end
+
+	local function OnRelease(self)
+		self:SetText("")
+		self:SetLabel("")
+		self:SetDisabled(false)
+
+		self.value = nil
+		self.list = nil
+		self.open = nil
+		self.hasClose = nil
+
+		self.frame:ClearAllPoints()
+		self.frame:Hide()
+	end
+
+	local function SetValue(self, value) -- Set the value to an item in the List.
+		if self.list then
+			self:SetText(value or "")
+		end
+		self.value = value
+	end
+
+	local function GetValue(self)
+		return self.value
+	end
+
+	local function SetList(self, list) -- Set the list of values for the dropdown (key => value pairs)
+		self.list = list or Media:HashTable("font")
+	end
+
+	local function SetText(self, text) -- Set the text displayed in the box.
+		self.frame.text:SetText(text or "")
+		local font = self.list[text] ~= text and self.list[text] or Media:Fetch('font',text)
+		local _, size, outline= self.frame.text:GetFont()
+		self.frame.text:SetFont(font,size,outline)
+	end
+
+	local function SetLabel(self, text) -- Set the text for the label.
+		self.frame.label:SetText(text or "")
+	end
+
+	local function AddItem(self, key, value) -- Add an item to the list.
+		self.list = self.list or {}
+		self.list[key] = value
+	end
+	local SetItemValue = AddItem -- Set the value of a item in the list. <<same as adding a new item>>
+
+	local function SetMultiselect(self, flag) end -- Toggle multi-selecting. <<Dummy function to stay inline with the dropdown API>>
+	local function GetMultiselect() return false end-- Query the multi-select flag. <<Dummy function to stay inline with the dropdown API>>
+	local function SetItemDisabled(self, key) end-- Disable one item in the list. <<Dummy function to stay inline with the dropdown API>>
+
+	local function SetDisabled(self, disabled) -- Disable the widget.
+		self.disabled = disabled
+		if disabled then
+			self.frame:Disable()
+		else
+			self.frame:Enable()
+		end
+	end
+
+	local function textSort(a,b)
+		return string.upper(a) < string.upper(b)
+	end
+
+	local sortedlist = {}
+	local function ToggleDrop(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+			AceGUI:ClearFocus()
+		else
+			AceGUI:SetFocus(self)
+			self.dropdown = AGSMW:GetDropDownFrame()
+			local width = self.frame:GetWidth()
+			self.dropdown:SetPoint("TOPLEFT", self.frame, "BOTTOMLEFT")
+			self.dropdown:SetPoint("TOPRIGHT", self.frame, "BOTTOMRIGHT", width < 160 and (160 - width) or 0, 0)
+			for k, v in pairs(self.list) do
+				sortedlist[#sortedlist+1] = k
+			end
+			table.sort(sortedlist, textSort)
+			for i, k in ipairs(sortedlist) do
+				local f = GetContentLine()
+				local _, size, outline= f.text:GetFont()
+				local font = self.list[k] ~= k and self.list[k] or Media:Fetch('font',k)
+				f.text:SetFont(font,size,outline)
+				f.text:SetText(k)
+				if k == self.value then
+					f.check:Show()
+				end
+				f.obj = self
+				self.dropdown:AddFrame(f)
+			end
+			wipe(sortedlist)
+		end
+	end
+
+	local function ClearFocus(self)
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function OnHide(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function Drop_OnEnter(this)
+		this.obj:Fire("OnEnter")
+	end
+
+	local function Drop_OnLeave(this)
+		this.obj:Fire("OnLeave")
+	end
+
+	local function Constructor()
+		local frame = AGSMW:GetBaseFrame()
+		local self = {}
+
+		self.type = widgetType
+		self.frame = frame
+		frame.obj = self
+		frame.dropButton.obj = self
+		frame.dropButton:SetScript("OnEnter", Drop_OnEnter)
+		frame.dropButton:SetScript("OnLeave", Drop_OnLeave)
+		frame.dropButton:SetScript("OnClick",ToggleDrop)
+		frame:SetScript("OnHide", OnHide)
+
+		self.alignoffset = 31
+
+		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.ToggleDrop = ToggleDrop
+
+		AceGUI:RegisterAsWidget(self)
+		return self
+	end
+
+	AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion)
+
+end
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets/SoundWidget.lua b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/SoundWidget.lua
new file mode 100644
index 0000000..60904a7
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/SoundWidget.lua
@@ -0,0 +1,264 @@
+-- Widget is based on the AceGUIWidget-DropDown.lua supplied with AceGUI-3.0
+-- Widget created by Yssaril
+
+local AceGUI = LibStub("AceGUI-3.0")
+local Media = LibStub("LibSharedMedia-3.0")
+
+local AGSMW = LibStub("AceGUISharedMediaWidgets-1.0")
+
+do
+	local widgetType = "LSM30_Sound"
+	local widgetVersion = 11
+
+	local contentFrameCache = {}
+	local function ReturnSelf(self)
+		self:ClearAllPoints()
+		self:Hide()
+		self.check:Hide()
+		table.insert(contentFrameCache, self)
+	end
+
+	local function ContentOnClick(this, button)
+		local self = this.obj
+		self:Fire("OnValueChanged", this.text:GetText())
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function ContentSpeakerOnClick(this, button)
+		local self = this.frame.obj
+		local sound = this.frame.text:GetText()
+		PlaySoundFile(self.list[sound] ~= sound and self.list[sound] or Media:Fetch('sound',sound), "Master")
+	end
+
+	local function GetContentLine()
+		local frame
+		if next(contentFrameCache) then
+			frame = table.remove(contentFrameCache)
+		else
+			frame = CreateFrame("Button", nil, UIParent)
+				--frame:SetWidth(200)
+				frame:SetHeight(18)
+				frame:SetHighlightTexture([[Interface\QuestFrame\UI-QuestTitleHighlight]], "ADD")
+				frame:SetScript("OnClick", ContentOnClick)
+			local check = frame:CreateTexture("OVERLAY")
+				check:SetWidth(16)
+				check:SetHeight(16)
+				check:SetPoint("LEFT",frame,"LEFT",1,-1)
+				check:SetTexture("Interface\\Buttons\\UI-CheckBox-Check")
+				check:Hide()
+			frame.check = check
+
+			local soundbutton = CreateFrame("Button", nil, frame)
+				soundbutton:SetWidth(16)
+				soundbutton:SetHeight(16)
+				soundbutton:SetPoint("RIGHT",frame,"RIGHT",-1,0)
+				soundbutton.frame = frame
+				soundbutton:SetScript("OnClick", ContentSpeakerOnClick)
+			frame.soundbutton = soundbutton
+
+			local speaker = soundbutton:CreateTexture(nil, "BACKGROUND")
+				speaker:SetTexture("Interface\\Common\\VoiceChat-Speaker")
+				speaker:SetAllPoints(soundbutton)
+			frame.speaker = speaker
+			local speakeron = soundbutton:CreateTexture(nil, "HIGHLIGHT")
+				speakeron:SetTexture("Interface\\Common\\VoiceChat-On")
+				speakeron:SetAllPoints(soundbutton)
+			frame.speakeron = speakeron
+
+			local text = frame:CreateFontString(nil,"OVERLAY","GameFontWhite")
+				text:SetPoint("TOPLEFT", check, "TOPRIGHT", 1, 0)
+				text:SetPoint("BOTTOMRIGHT", soundbutton, "BOTTOMLEFT", -2, 0)
+				text:SetJustifyH("LEFT")
+				text:SetText("Test Test Test Test Test Test Test")
+			frame.text = text
+			frame.ReturnSelf = ReturnSelf
+		end
+		frame:Show()
+		return frame
+	end
+
+	local function OnAcquire(self)
+		self:SetHeight(44)
+		self:SetWidth(200)
+	end
+
+	local function OnRelease(self)
+		self:SetText("")
+		self:SetLabel("")
+		self:SetDisabled(false)
+
+		self.value = nil
+		self.list = nil
+		self.open = nil
+		self.hasClose = nil
+
+		self.frame:ClearAllPoints()
+		self.frame:Hide()
+	end
+
+	local function SetValue(self, value) -- Set the value to an item in the List.
+		if self.list then
+			self:SetText(value or "")
+		end
+		self.value = value
+	end
+
+	local function GetValue(self)
+		return self.value
+	end
+
+	local function SetList(self, list) -- Set the list of values for the dropdown (key => value pairs)
+		self.list = list or Media:HashTable("sound")
+	end
+
+	local function SetText(self, text) -- Set the text displayed in the box.
+		self.frame.text:SetText(text or "")
+	end
+
+	local function SetLabel(self, text) -- Set the text for the label.
+		self.frame.label:SetText(text or "")
+	end
+
+	local function AddItem(self, key, value) -- Add an item to the list.
+		self.list = self.list or {}
+		self.list[key] = value
+	end
+	local SetItemValue = AddItem -- Set the value of a item in the list. <<same as adding a new item>>
+
+	local function SetMultiselect(self, flag) end -- Toggle multi-selecting. <<Dummy function to stay inline with the dropdown API>>
+	local function GetMultiselect() return false end-- Query the multi-select flag. <<Dummy function to stay inline with the dropdown API>>
+	local function SetItemDisabled(self, key) end-- Disable one item in the list. <<Dummy function to stay inline with the dropdown API>>
+
+	local function SetDisabled(self, disabled) -- Disable the widget.
+		self.disabled = disabled
+		if disabled then
+			self.frame:Disable()
+			self.speaker:SetDesaturated(true)
+			self.speakeron:SetDesaturated(true)
+		else
+			self.frame:Enable()
+			self.speaker:SetDesaturated(false)
+			self.speakeron:SetDesaturated(false)
+		end
+	end
+
+	local function textSort(a,b)
+		return string.upper(a) < string.upper(b)
+	end
+
+	local sortedlist = {}
+	local function ToggleDrop(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+			AceGUI:ClearFocus()
+		else
+			AceGUI:SetFocus(self)
+			self.dropdown = AGSMW:GetDropDownFrame()
+			local width = self.frame:GetWidth()
+			self.dropdown:SetPoint("TOPLEFT", self.frame, "BOTTOMLEFT")
+			self.dropdown:SetPoint("TOPRIGHT", self.frame, "BOTTOMRIGHT", width < 160 and (160 - width) or 0, 0)
+			for k, v in pairs(self.list) do
+				sortedlist[#sortedlist+1] = k
+			end
+			table.sort(sortedlist, textSort)
+			for i, k in ipairs(sortedlist) do
+				local f = GetContentLine()
+				f.text:SetText(k)
+				if k == self.value then
+					f.check:Show()
+				end
+				f.obj = self
+				self.dropdown:AddFrame(f)
+			end
+			wipe(sortedlist)
+		end
+	end
+
+	local function ClearFocus(self)
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function OnHide(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function Drop_OnEnter(this)
+		this.obj:Fire("OnEnter")
+	end
+
+	local function Drop_OnLeave(this)
+		this.obj:Fire("OnLeave")
+	end
+
+	local function WidgetPlaySound(this)
+		local self = this.obj
+		local sound = self.frame.text:GetText()
+		PlaySoundFile(self.list[sound] ~= sound and self.list[sound] or Media:Fetch('sound',sound), "Master")
+	end
+
+	local function Constructor()
+		local frame = AGSMW:GetBaseFrame()
+		local self = {}
+
+		self.type = widgetType
+		self.frame = frame
+		frame.obj = self
+		frame.dropButton.obj = self
+		frame.dropButton:SetScript("OnEnter", Drop_OnEnter)
+		frame.dropButton:SetScript("OnLeave", Drop_OnLeave)
+		frame.dropButton:SetScript("OnClick",ToggleDrop)
+		frame:SetScript("OnHide", OnHide)
+
+
+		local soundbutton = CreateFrame("Button", nil, frame)
+			soundbutton:SetWidth(16)
+			soundbutton:SetHeight(16)
+			soundbutton:SetPoint("LEFT",frame.DLeft,"LEFT",26,1)
+			soundbutton:SetScript("OnClick", WidgetPlaySound)
+			soundbutton.obj = self
+		self.soundbutton = soundbutton
+		frame.text:SetPoint("LEFT",soundbutton,"RIGHT",2,0)
+
+
+		local speaker = soundbutton:CreateTexture(nil, "BACKGROUND")
+			speaker:SetTexture("Interface\\Common\\VoiceChat-Speaker")
+			speaker:SetAllPoints(soundbutton)
+		self.speaker = speaker
+		local speakeron = soundbutton:CreateTexture(nil, "HIGHLIGHT")
+			speakeron:SetTexture("Interface\\Common\\VoiceChat-On")
+			speakeron:SetAllPoints(soundbutton)
+		self.speakeron = speakeron
+
+		self.alignoffset = 31
+
+		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.ToggleDrop = ToggleDrop
+
+		AceGUI:RegisterAsWidget(self)
+		return self
+	end
+
+	AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion)
+
+end
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets/StatusbarWidget.lua b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/StatusbarWidget.lua
new file mode 100644
index 0000000..98b85f5
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/StatusbarWidget.lua
@@ -0,0 +1,233 @@
+-- Widget is based on the AceGUIWidget-DropDown.lua supplied with AceGUI-3.0
+-- Widget created by Yssaril
+
+local AceGUI = LibStub("AceGUI-3.0")
+local Media = LibStub("LibSharedMedia-3.0")
+
+local AGSMW = LibStub("AceGUISharedMediaWidgets-1.0")
+
+do
+	local widgetType = "LSM30_Statusbar"
+	local widgetVersion = 11
+
+	local contentFrameCache = {}
+	local function ReturnSelf(self)
+		self:ClearAllPoints()
+		self:Hide()
+		self.check:Hide()
+		table.insert(contentFrameCache, self)
+	end
+
+	local function ContentOnClick(this, button)
+		local self = this.obj
+		self:Fire("OnValueChanged", this.text:GetText())
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function GetContentLine()
+		local frame
+		if next(contentFrameCache) then
+			frame = table.remove(contentFrameCache)
+		else
+			frame = CreateFrame("Button", nil, UIParent)
+				--frame:SetWidth(200)
+				frame:SetHeight(18)
+				frame:SetHighlightTexture([[Interface\QuestFrame\UI-QuestTitleHighlight]], "ADD")
+				frame:SetScript("OnClick", ContentOnClick)
+			local check = frame:CreateTexture("OVERLAY")
+				check:SetWidth(16)
+				check:SetHeight(16)
+				check:SetPoint("LEFT",frame,"LEFT",1,-1)
+				check:SetTexture("Interface\\Buttons\\UI-CheckBox-Check")
+				check:Hide()
+			frame.check = check
+			local bar = frame:CreateTexture("ARTWORK")
+				bar:SetHeight(16)
+				bar:SetPoint("LEFT",check,"RIGHT",1,0)
+				bar:SetPoint("RIGHT",frame,"RIGHT",-1,0)
+			frame.bar = bar
+			local text = frame:CreateFontString(nil,"OVERLAY","GameFontWhite")
+
+				local font, size = text:GetFont()
+				text:SetFont(font,size,"OUTLINE")
+
+				text:SetPoint("TOPLEFT", check, "TOPRIGHT", 3, 0)
+				text:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT", -2, 0)
+				text:SetJustifyH("LEFT")
+				text:SetText("Test Test Test Test Test Test Test")
+			frame.text = text
+			frame.ReturnSelf = ReturnSelf
+		end
+		frame:Show()
+		return frame
+	end
+
+	local function OnAcquire(self)
+		self:SetHeight(44)
+		self:SetWidth(200)
+	end
+
+	local function OnRelease(self)
+		self:SetText("")
+		self:SetLabel("")
+		self:SetDisabled(false)
+
+		self.value = nil
+		self.list = nil
+		self.open = nil
+		self.hasClose = nil
+
+		self.frame:ClearAllPoints()
+		self.frame:Hide()
+	end
+
+	local function SetValue(self, value) -- Set the value to an item in the List.
+		if self.list then
+			self:SetText(value or "")
+		end
+		self.value = value
+	end
+
+	local function GetValue(self)
+		return self.value
+	end
+
+	local function SetList(self, list) -- Set the list of values for the dropdown (key => value pairs)
+		self.list = list or Media:HashTable("statusbar")
+	end
+
+
+	local function SetText(self, text) -- Set the text displayed in the box.
+		self.frame.text:SetText(text or "")
+		local statusbar = self.list[text] ~= text and self.list[text] or Media:Fetch('statusbar',text)
+		self.bar:SetTexture(statusbar)
+	end
+
+	local function SetLabel(self, text) -- Set the text for the label.
+		self.frame.label:SetText(text or "")
+	end
+
+	local function AddItem(self, key, value) -- Add an item to the list.
+		self.list = self.list or {}
+		self.list[key] = value
+	end
+	local SetItemValue = AddItem -- Set the value of a item in the list. <<same as adding a new item>>
+
+	local function SetMultiselect(self, flag) end -- Toggle multi-selecting. <<Dummy function to stay inline with the dropdown API>>
+	local function GetMultiselect() return false end-- Query the multi-select flag. <<Dummy function to stay inline with the dropdown API>>
+	local function SetItemDisabled(self, key) end-- Disable one item in the list. <<Dummy function to stay inline with the dropdown API>>
+
+	local function SetDisabled(self, disabled) -- Disable the widget.
+		self.disabled = disabled
+		if disabled then
+			self.frame:Disable()
+		else
+			self.frame:Enable()
+		end
+	end
+
+	local function textSort(a,b)
+		return string.upper(a) < string.upper(b)
+	end
+
+	local sortedlist = {}
+	local function ToggleDrop(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+			AceGUI:ClearFocus()
+		else
+			AceGUI:SetFocus(self)
+			self.dropdown = AGSMW:GetDropDownFrame()
+			local width = self.frame:GetWidth()
+			self.dropdown:SetPoint("TOPLEFT", self.frame, "BOTTOMLEFT")
+			self.dropdown:SetPoint("TOPRIGHT", self.frame, "BOTTOMRIGHT", width < 160 and (160 - width) or 0, 0)
+			for k, v in pairs(self.list) do
+				sortedlist[#sortedlist+1] = k
+			end
+			table.sort(sortedlist, textSort)
+			for i, k in ipairs(sortedlist) do
+				local f = GetContentLine()
+				f.text:SetText(k)
+				--print(k)
+				if k == self.value then
+					f.check:Show()
+				end
+
+				local statusbar = self.list[k] ~= k and self.list[k] or Media:Fetch('statusbar',k)
+				f.bar:SetTexture(statusbar)
+				f.obj = self
+				f.dropdown = self.dropdown
+				self.dropdown:AddFrame(f)
+			end
+			wipe(sortedlist)
+		end
+	end
+
+	local function ClearFocus(self)
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function OnHide(this)
+		local self = this.obj
+		if self.dropdown then
+			self.dropdown = AGSMW:ReturnDropDownFrame(self.dropdown)
+		end
+	end
+
+	local function Drop_OnEnter(this)
+		this.obj:Fire("OnEnter")
+	end
+
+	local function Drop_OnLeave(this)
+		this.obj:Fire("OnLeave")
+	end
+
+	local function Constructor()
+		local frame = AGSMW:GetBaseFrame()
+		local self = {}
+
+		self.type = widgetType
+		self.frame = frame
+		frame.obj = self
+		frame.dropButton.obj = self
+		frame.dropButton:SetScript("OnEnter", Drop_OnEnter)
+		frame.dropButton:SetScript("OnLeave", Drop_OnLeave)
+		frame.dropButton:SetScript("OnClick",ToggleDrop)
+		frame:SetScript("OnHide", OnHide)
+
+		local bar = frame:CreateTexture(nil, "OVERLAY")
+			bar:SetPoint("TOPLEFT", frame,"TOPLEFT",6,-25)
+			bar:SetPoint("BOTTOMRIGHT", frame,"BOTTOMRIGHT", -21, 5)
+			bar:SetAlpha(0.5)
+		self.bar = bar
+
+		self.alignoffset = 31
+
+		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.ToggleDrop = ToggleDrop
+
+		AceGUI:RegisterAsWidget(self)
+		return self
+	end
+
+	AceGUI:RegisterWidgetType(widgetType, Constructor, widgetVersion)
+
+end
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets/prototypes.lua b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/prototypes.lua
new file mode 100644
index 0000000..cd7faf9
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/prototypes.lua
@@ -0,0 +1,273 @@
+-- Widget created by Yssaril
+--[===[@debug@
+local DataVersion = 9001 -- dev version always overwrites everything else :)
+--@end-debug@]===]
+--@non-debug@
+local DataVersion = 56
+--@end-non-debug@
+local AGSMW = LibStub:NewLibrary("AceGUISharedMediaWidgets-1.0", DataVersion)
+
+if not AGSMW then
+  return	-- already loaded and no upgrade necessary
+end
+
+local AceGUI = LibStub("AceGUI-3.0")
+local Media = LibStub("LibSharedMedia-3.0")
+
+AGSMW = AGSMW or {}
+
+AceGUIWidgetLSMlists = {
+	['font'] = Media:HashTable("font"),
+	['sound'] = Media:HashTable("sound"),
+	['statusbar'] = Media:HashTable("statusbar"),
+	['border'] = Media:HashTable("border"),
+	['background'] = Media:HashTable("background"),
+}
+
+do
+	local function disable(frame)
+		frame.label:SetTextColor(.5,.5,.5)
+		frame.text:SetTextColor(.5,.5,.5)
+		frame.dropButton:Disable()
+		if frame.displayButtonFont then
+			frame.displayButtonFont:SetTextColor(.5,.5,.5)
+			frame.displayButton:Disable()
+		end
+	end
+
+	local function enable(frame)
+		frame.label:SetTextColor(1,.82,0)
+		frame.text:SetTextColor(1,1,1)
+		frame.dropButton:Enable()
+		if frame.displayButtonFont then
+			frame.displayButtonFont:SetTextColor(1,1,1)
+			frame.displayButton:Enable()
+		end
+	end
+
+	local displayButtonBackdrop = {
+		edgeFile = "Interface/Tooltips/UI-Tooltip-Border",
+		tile = true, tileSize = 16, edgeSize = 16,
+		insets = { left = 4, right = 4, top = 4, bottom = 4 },
+	}
+
+	-- create or retrieve BaseFrame
+	function AGSMW:GetBaseFrame()
+		local frame = CreateFrame("Frame", nil, UIParent)
+		frame:SetHeight(44)
+		frame:SetWidth(200)
+
+		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:SetText("")
+		frame.label = label
+
+		local DLeft = frame:CreateTexture(nil, "ARTWORK")
+			DLeft:SetWidth(25)
+			DLeft:SetHeight(64)
+			DLeft:SetPoint("BOTTOMLEFT", frame, "BOTTOMLEFT", -17, -21)
+			DLeft:SetTexture([[Interface\Glues\CharacterCreate\CharacterCreate-LabelFrame]])
+			DLeft:SetTexCoord(0, 0.1953125, 0, 1)
+		frame.DLeft = DLeft
+
+		local DRight = frame:CreateTexture(nil, "ARTWORK")
+			DRight:SetWidth(25)
+			DRight:SetHeight(64)
+			DRight:SetPoint("TOP", DLeft, "TOP")
+			DRight:SetPoint("RIGHT", frame, "RIGHT", 17, 0)
+			DRight:SetTexture([[Interface\Glues\CharacterCreate\CharacterCreate-LabelFrame]])
+			DRight:SetTexCoord(0.8046875, 1, 0, 1)
+		frame.DRight = DRight
+
+		local DMiddle = frame:CreateTexture(nil, "ARTWORK")
+			DMiddle:SetHeight(64)
+			DMiddle:SetPoint("TOP", DLeft, "TOP")
+			DMiddle:SetPoint("LEFT", DLeft, "RIGHT")
+			DMiddle:SetPoint("RIGHT", DRight, "LEFT")
+			DMiddle:SetTexture([[Interface\Glues\CharacterCreate\CharacterCreate-LabelFrame]])
+			DMiddle:SetTexCoord(0.1953125, 0.8046875, 0, 1)
+		frame.DMiddle = DMiddle
+
+		local text = frame:CreateFontString(nil,"OVERLAY","GameFontHighlightSmall")
+			text:SetPoint("RIGHT",DRight,"RIGHT",-43,1)
+			text:SetPoint("LEFT",DLeft,"LEFT",26,1)
+			text:SetJustifyH("RIGHT")
+			text:SetHeight(18)
+			text:SetText("")
+		frame.text = text
+
+		local dropButton = CreateFrame("Button", nil, frame)
+			dropButton:SetWidth(24)
+			dropButton:SetHeight(24)
+			dropButton:SetPoint("TOPRIGHT", DRight, "TOPRIGHT", -16, -18)
+			dropButton:SetNormalTexture([[Interface\ChatFrame\UI-ChatIcon-ScrollDown-Up]])
+			dropButton:SetPushedTexture([[Interface\ChatFrame\UI-ChatIcon-ScrollDown-Down]])
+			dropButton:SetDisabledTexture([[Interface\ChatFrame\UI-ChatIcon-ScrollDown-Disabled]])
+			dropButton:SetHighlightTexture([[Interface\Buttons\UI-Common-MouseHilight]], "ADD")
+		frame.dropButton = dropButton
+
+		frame.Disable = disable
+		frame.Enable = enable
+		return frame
+	end
+
+	function AGSMW:GetBaseFrameWithWindow()
+		local frame = self:GetBaseFrame()
+
+		local displayButton = CreateFrame("Button", nil, frame)
+			displayButton:SetHeight(42)
+			displayButton:SetWidth(42)
+			displayButton:SetPoint("TOPLEFT", frame, "TOPLEFT", 1, -2)
+			displayButton:SetBackdrop(displayButtonBackdrop)
+			displayButton:SetBackdropBorderColor(.5, .5, .5)
+		frame.displayButton = displayButton
+
+		frame.label:SetPoint("TOPLEFT",displayButton,"TOPRIGHT",1,2)
+
+		frame.DLeft:SetPoint("BOTTOMLEFT", displayButton, "BOTTOMRIGHT", -17, -20)
+
+		return frame
+	end
+
+end
+
+do
+
+	local sliderBackdrop = {
+		["bgFile"] = [[Interface\Buttons\UI-SliderBar-Background]],
+		["edgeFile"] = [[Interface\Buttons\UI-SliderBar-Border]],
+		["tile"] = true,
+		["edgeSize"] = 8,
+		["tileSize"] = 8,
+		["insets"] = {
+			["left"] = 3,
+			["right"] = 3,
+			["top"] = 3,
+			["bottom"] = 3,
+		},
+	}
+	local frameBackdrop = {
+		bgFile=[[Interface\DialogFrame\UI-DialogBox-Background-Dark]],
+		edgeFile = [[Interface\DialogFrame\UI-DialogBox-Border]],
+		tile = true, tileSize = 32, edgeSize = 32,
+		insets = { left = 11, right = 12, top = 12, bottom = 9 },
+	}
+
+	local function OnMouseWheel(self, dir)
+		self.slider:SetValue(self.slider:GetValue()+(15*dir*-1))
+	end
+
+	local function AddFrame(self, frame)
+		frame:SetParent(self.contentframe)
+		frame:SetFrameStrata(self:GetFrameStrata())
+		frame:SetFrameLevel(self:GetFrameLevel() + 100)
+
+		if next(self.contentRepo) then
+			frame:SetPoint("TOPLEFT", self.contentRepo[#self.contentRepo], "BOTTOMLEFT", 0, 0)
+			frame:SetPoint("RIGHT", self.contentframe, "RIGHT", 0, 0)
+			self.contentframe:SetHeight(self.contentframe:GetHeight() + frame:GetHeight())
+			self.contentRepo[#self.contentRepo+1] = frame
+		else
+			self.contentframe:SetHeight(frame:GetHeight())
+			frame:SetPoint("TOPLEFT", self.contentframe, "TOPLEFT", 0, 0)
+			frame:SetPoint("RIGHT", self.contentframe, "RIGHT", 0, 0)
+			self.contentRepo[1] = frame
+		end
+
+		if self.contentframe:GetHeight() > UIParent:GetHeight()*2/5 - 20 then
+			self.scrollframe:SetPoint("BOTTOMRIGHT", self, "BOTTOMRIGHT", -28, 12)
+			self:SetHeight(UIParent:GetHeight()*2/5)
+			self.slider:Show()
+			self:SetScript("OnMouseWheel", OnMouseWheel)
+			self.scrollframe:UpdateScrollChildRect()
+			self.slider:SetMinMaxValues(0, self.contentframe:GetHeight()-self.scrollframe:GetHeight())
+		else
+			self.scrollframe:SetPoint("BOTTOMRIGHT", self, "BOTTOMRIGHT", -14, 12)
+			self:SetHeight(self.contentframe:GetHeight()+25)
+			self.slider:Hide()
+			self:SetScript("OnMouseWheel", nil)
+			self.scrollframe:UpdateScrollChildRect()
+			self.slider:SetMinMaxValues(0, 0)
+		end
+		self.contentframe:SetWidth(self.scrollframe:GetWidth())
+	end
+
+	local function ClearFrames(self)
+		for i, frame in ipairs(self.contentRepo) do
+			frame:ReturnSelf()
+			self.contentRepo[i] = nil
+		end
+	end
+
+	local function slider_OnValueChanged(self, value)
+		self.frame.scrollframe:SetVerticalScroll(value)
+	end
+
+	local DropDownCache = {}
+	function AGSMW:GetDropDownFrame()
+		local frame
+		if next(DropDownCache) then
+			frame = table.remove(DropDownCache)
+		else
+			frame = CreateFrame("Frame", nil, UIParent)
+				frame:SetClampedToScreen(true)
+				frame:SetWidth(188)
+				frame:SetBackdrop(frameBackdrop)
+				frame:SetFrameStrata("TOOLTIP")
+				frame:EnableMouseWheel(true)
+
+			local contentframe = CreateFrame("Frame", nil, frame)
+				contentframe:SetWidth(160)
+				contentframe:SetHeight(0)
+			frame.contentframe = contentframe
+
+			local scrollframe = CreateFrame("ScrollFrame", nil, frame)
+				scrollframe:SetWidth(160)
+				scrollframe:SetPoint("TOPLEFT", frame, "TOPLEFT", 14, -13)
+				scrollframe:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT", -14, 12)
+				scrollframe:SetScrollChild(contentframe)
+			frame.scrollframe = scrollframe
+
+			contentframe:SetPoint("TOPLEFT", scrollframe)
+			contentframe:SetPoint("TOPRIGHT", scrollframe)
+
+			local bgTex = frame:CreateTexture(nil, "ARTWORK")
+				bgTex:SetAllPoints(scrollframe)
+			frame.bgTex = bgTex
+
+			frame.AddFrame = AddFrame
+			frame.ClearFrames = ClearFrames
+			frame.contentRepo = {} -- store all our frames in here so we can get rid of them later
+
+			local slider = CreateFrame("Slider", nil, scrollframe)
+				slider:SetOrientation("VERTICAL")
+				slider:SetPoint("TOPRIGHT", frame, "TOPRIGHT", -14, -10)
+				slider:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT", -14, 10)
+				slider:SetBackdrop(sliderBackdrop)
+				slider:SetThumbTexture([[Interface\Buttons\UI-SliderBar-Button-Vertical]])
+				slider:SetMinMaxValues(0, 1)
+				--slider:SetValueStep(1)
+				slider:SetWidth(12)
+				slider.frame = frame
+				slider:SetScript("OnValueChanged", slider_OnValueChanged)
+			frame.slider = slider
+		end
+		frame:SetHeight(UIParent:GetHeight()*2/5)
+		frame.slider:SetValue(0)
+		frame:Show()
+		return frame
+	end
+
+	function AGSMW:ReturnDropDownFrame(frame)
+		ClearFrames(frame)
+		frame:ClearAllPoints()
+		frame:Hide()
+		frame:SetBackdrop(frameBackdrop)
+		frame.bgTex:SetTexture(nil)
+		table.insert(DropDownCache, frame)
+		return nil
+	end
+end
diff --git a/Titan/libs/AceGUI-3.0-SharedMediaWidgets/widget.xml b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/widget.xml
new file mode 100644
index 0000000..798546f
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0-SharedMediaWidgets/widget.xml
@@ -0,0 +1,9 @@
+<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="prototypes.lua" />
+	<Script file="FontWidget.lua" />
+	<Script file="SoundWidget.lua" />
+	<Script file="StatusbarWidget.lua" />
+	<Script file="BorderWidget.lua" />
+	<Script file="BackgroundWidget.lua" />
+</Ui>
\ No newline at end of file
diff --git a/Titan/libs/AceGUI-3.0/AceGUI-3.0.lua b/Titan/libs/AceGUI-3.0/AceGUI-3.0.lua
new file mode 100755
index 0000000..fcc4e94
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/AceGUI-3.0.lua
@@ -0,0 +1,1031 @@
+--- **AceGUI-3.0** provides access to numerous widgets which can be used to create GUIs.
+-- AceGUI is used by AceConfigDialog to create the option GUIs, but you can use it by itself
+-- to create any custom GUI. There are more extensive examples in the test suite in the Ace3
+-- stand-alone distribution.
+--
+-- **Note**: When using AceGUI-3.0 directly, please do not modify the frames of the widgets directly,
+-- as any "unknown" change to the widgets will cause addons that get your widget out of the widget pool
+-- to misbehave. If you think some part of a widget should be modifiable, please open a ticket, and we"ll
+-- implement a proper API to modify it.
+-- @usage
+-- local AceGUI = LibStub("AceGUI-3.0")
+-- -- Create a container frame
+-- local f = AceGUI:Create("Frame")
+-- f:SetCallback("OnClose",function(widget) AceGUI:Release(widget) end)
+-- f:SetTitle("AceGUI-3.0 Example")
+-- f:SetStatusText("Status Bar")
+-- f:SetLayout("Flow")
+-- -- Create a button
+-- local btn = AceGUI:Create("Button")
+-- btn:SetWidth(170)
+-- btn:SetText("Button !")
+-- btn:SetCallback("OnClick", function() print("Click!") end)
+-- -- Add the button to the container
+-- f:AddChild(btn)
+-- @class file
+-- @name AceGUI-3.0
+-- @release $Id: AceGUI-3.0.lua 1177 2018-06-25 12:12:48Z nevcairiel $
+local ACEGUI_MAJOR, ACEGUI_MINOR = "AceGUI-3.0", 36
+local AceGUI, oldminor = LibStub:NewLibrary(ACEGUI_MAJOR, ACEGUI_MINOR)
+
+if not AceGUI then return end -- No upgrade needed
+
+-- Lua APIs
+local tconcat, tremove, tinsert = table.concat, table.remove, table.insert
+local select, pairs, next, type = select, pairs, next, type
+local error, assert, loadstring = error, assert, loadstring
+local setmetatable, rawget, rawset = setmetatable, rawget, rawset
+local math_max = math.max
+
+-- WoW APIs
+local UIParent = 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: geterrorhandler, LibStub
+
+--local con = LibStub("AceConsole-3.0",true)
+
+AceGUI.WidgetRegistry = AceGUI.WidgetRegistry or {}
+AceGUI.LayoutRegistry = AceGUI.LayoutRegistry or {}
+AceGUI.WidgetBase = AceGUI.WidgetBase or {}
+AceGUI.WidgetContainerBase = AceGUI.WidgetContainerBase or {}
+AceGUI.WidgetVersions = AceGUI.WidgetVersions or {}
+
+-- local upvalues
+local WidgetRegistry = AceGUI.WidgetRegistry
+local LayoutRegistry = AceGUI.LayoutRegistry
+local WidgetVersions = AceGUI.WidgetVersions
+
+--[[
+	 xpcall safecall implementation
+]]
+local xpcall = xpcall
+
+local function errorhandler(err)
+	return geterrorhandler()(err)
+end
+
+local function CreateDispatcher(argCount)
+	local code = [[
+		local xpcall, eh = ...
+		local method, ARGS
+		local function call() return method(ARGS) end
+
+		local function dispatch(func, ...)
+			method = func
+			if not method then return end
+			ARGS = ...
+			return xpcall(call, eh)
+		end
+
+		return dispatch
+	]]
+
+	local ARGS = {}
+	for i = 1, argCount do ARGS[i] = "arg"..i end
+	code = code:gsub("ARGS", tconcat(ARGS, ", "))
+	return assert(loadstring(code, "safecall Dispatcher["..argCount.."]"))(xpcall, errorhandler)
+end
+
+local Dispatchers = setmetatable({}, {__index=function(self, argCount)
+	local dispatcher = CreateDispatcher(argCount)
+	rawset(self, argCount, dispatcher)
+	return dispatcher
+end})
+Dispatchers[0] = function(func)
+	return xpcall(func, errorhandler)
+end
+
+local function safecall(func, ...)
+	return Dispatchers[select("#", ...)](func, ...)
+end
+
+-- Recycling functions
+local newWidget, delWidget
+do
+	-- Version Upgrade in Minor 29
+	-- Internal Storage of the objects changed, from an array table
+	-- to a hash table, and additionally we introduced versioning on
+	-- the widgets which would discard all widgets from a pre-29 version
+	-- anyway, so we just clear the storage now, and don't try to
+	-- convert the storage tables to the new format.
+	-- This should generally not cause *many* widgets to end up in trash,
+	-- since once dialogs are opened, all addons should be loaded already
+	-- and AceGUI should be on the latest version available on the users
+	-- setup.
+	-- -- nevcairiel - Nov 2nd, 2009
+	if oldminor and oldminor < 29 and AceGUI.objPools then
+		AceGUI.objPools = nil
+	end
+
+	AceGUI.objPools = AceGUI.objPools or {}
+	local objPools = AceGUI.objPools
+	--Returns a new instance, if none are available either returns a new table or calls the given contructor
+	function newWidget(type)
+		if not WidgetRegistry[type] then
+			error("Attempt to instantiate unknown widget type", 2)
+		end
+
+		if not objPools[type] then
+			objPools[type] = {}
+		end
+
+		local newObj = next(objPools[type])
+		if not newObj then
+			newObj = WidgetRegistry[type]()
+			newObj.AceGUIWidgetVersion = WidgetVersions[type]
+		else
+			objPools[type][newObj] = nil
+			-- if the widget is older then the latest, don't even try to reuse it
+			-- just forget about it, and grab a new one.
+			if not newObj.AceGUIWidgetVersion or newObj.AceGUIWidgetVersion < WidgetVersions[type] then
+				return newWidget(type)
+			end
+		end
+		return newObj
+	end
+	-- Releases an instance to the Pool
+	function delWidget(obj,type)
+		if not objPools[type] then
+			objPools[type] = {}
+		end
+		if objPools[type][obj] then
+			error("Attempt to Release Widget that is already released", 2)
+		end
+		objPools[type][obj] = true
+	end
+end
+
+
+-------------------
+-- API Functions --
+-------------------
+
+-- Gets a widget Object
+
+--- Create a new Widget of the given type.
+-- This function will instantiate a new widget (or use one from the widget pool), and call the
+-- OnAcquire function on it, before returning.
+-- @param type The type of the widget.
+-- @return The newly created widget.
+function AceGUI:Create(type)
+	if WidgetRegistry[type] then
+		local widget = newWidget(type)
+
+		if rawget(widget, "Acquire") then
+			widget.OnAcquire = widget.Acquire
+			widget.Acquire = nil
+		elseif rawget(widget, "Aquire") then
+			widget.OnAcquire = widget.Aquire
+			widget.Aquire = nil
+		end
+
+		if rawget(widget, "Release") then
+			widget.OnRelease = rawget(widget, "Release")
+			widget.Release = nil
+		end
+
+		if widget.OnAcquire then
+			widget:OnAcquire()
+		else
+			error(("Widget type %s doesn't supply an OnAcquire Function"):format(type))
+		end
+		-- Set the default Layout ("List")
+		safecall(widget.SetLayout, widget, "List")
+		safecall(widget.ResumeLayout, widget)
+		return widget
+	end
+end
+
+--- Releases a widget Object.
+-- This function calls OnRelease on the widget and places it back in the widget pool.
+-- Any data on the widget is being erased, and the widget will be hidden.\\
+-- If this widget is a Container-Widget, all of its Child-Widgets will be releases as well.
+-- @param widget The widget to release
+function AceGUI:Release(widget)
+	safecall(widget.PauseLayout, widget)
+	widget:Fire("OnRelease")
+	safecall(widget.ReleaseChildren, widget)
+
+	if widget.OnRelease then
+		widget:OnRelease()
+--	else
+--		error(("Widget type %s doesn't supply an OnRelease Function"):format(widget.type))
+	end
+	for k in pairs(widget.userdata) do
+		widget.userdata[k] = nil
+	end
+	for k in pairs(widget.events) do
+		widget.events[k] = nil
+	end
+	widget.width = nil
+	widget.relWidth = nil
+	widget.height = nil
+	widget.relHeight = nil
+	widget.noAutoHeight = nil
+	widget.frame:ClearAllPoints()
+	widget.frame:Hide()
+	widget.frame:SetParent(UIParent)
+	widget.frame.width = nil
+	widget.frame.height = nil
+	if widget.content then
+		widget.content.width = nil
+		widget.content.height = nil
+	end
+	delWidget(widget, widget.type)
+end
+
+-----------
+-- Focus --
+-----------
+
+
+--- Called when a widget has taken focus.
+-- e.g. Dropdowns opening, Editboxes gaining kb focus
+-- @param widget The widget that should be focused
+function AceGUI:SetFocus(widget)
+	if self.FocusedWidget and self.FocusedWidget ~= widget then
+		safecall(self.FocusedWidget.ClearFocus, self.FocusedWidget)
+	end
+	self.FocusedWidget = widget
+end
+
+
+--- Called when something has happened that could cause widgets with focus to drop it
+-- e.g. titlebar of a frame being clicked
+function AceGUI:ClearFocus()
+	if self.FocusedWidget then
+		safecall(self.FocusedWidget.ClearFocus, self.FocusedWidget)
+		self.FocusedWidget = nil
+	end
+end
+
+-------------
+-- Widgets --
+-------------
+--[[
+	Widgets must provide the following functions
+		OnAcquire() - Called when the object is acquired, should set everything to a default hidden state
+
+	And the following members
+		frame - the frame or derivitive object that will be treated as the widget for size and anchoring purposes
+		type - the type of the object, same as the name given to :RegisterWidget()
+
+	Widgets contain a table called userdata, this is a safe place to store data associated with the wigdet
+	It will be cleared automatically when a widget is released
+	Placing values directly into a widget object should be avoided
+
+	If the Widget can act as a container for other Widgets the following
+		content - frame or derivitive that children will be anchored to
+
+	The Widget can supply the following Optional Members
+		:OnRelease() - Called when the object is Released, should remove any additional anchors and clear any data
+		:OnWidthSet(width) - Called when the width of the widget is changed
+		:OnHeightSet(height) - Called when the height of the widget is changed
+			Widgets should not use the OnSizeChanged events of thier frame or content members, use these methods instead
+			AceGUI already sets a handler to the event
+		:LayoutFinished(width, height) - called after a layout has finished, the width and height will be the width and height of the
+			area used for controls. These can be nil if the layout used the existing size to layout the controls.
+
+]]
+
+--------------------------
+-- Widget Base Template --
+--------------------------
+do
+	local WidgetBase = AceGUI.WidgetBase
+
+	WidgetBase.SetParent = function(self, parent)
+		local frame = self.frame
+		frame:SetParent(nil)
+		frame:SetParent(parent.content)
+		self.parent = parent
+	end
+
+	WidgetBase.SetCallback = function(self, name, func)
+		if type(func) == "function" then
+			self.events[name] = func
+		end
+	end
+
+	WidgetBase.Fire = function(self, name, ...)
+		if self.events[name] then
+			local success, ret = safecall(self.events[name], self, name, ...)
+			if success then
+				return ret
+			end
+		end
+	end
+
+	WidgetBase.SetWidth = function(self, width)
+		self.frame:SetWidth(width)
+		self.frame.width = width
+		if self.OnWidthSet then
+			self:OnWidthSet(width)
+		end
+	end
+
+	WidgetBase.SetRelativeWidth = function(self, width)
+		if width <= 0 or width > 1 then
+			error(":SetRelativeWidth(width): Invalid relative width.", 2)
+		end
+		self.relWidth = width
+		self.width = "relative"
+	end
+
+	WidgetBase.SetHeight = function(self, height)
+		self.frame:SetHeight(height)
+		self.frame.height = height
+		if self.OnHeightSet then
+			self:OnHeightSet(height)
+		end
+	end
+
+	--[[ WidgetBase.SetRelativeHeight = function(self, height)
+		if height <= 0 or height > 1 then
+			error(":SetRelativeHeight(height): Invalid relative height.", 2)
+		end
+		self.relHeight = height
+		self.height = "relative"
+	end ]]
+
+	WidgetBase.IsVisible = function(self)
+		return self.frame:IsVisible()
+	end
+
+	WidgetBase.IsShown= function(self)
+		return self.frame:IsShown()
+	end
+
+	WidgetBase.Release = function(self)
+		AceGUI:Release(self)
+	end
+
+	WidgetBase.SetPoint = function(self, ...)
+		return self.frame:SetPoint(...)
+	end
+
+	WidgetBase.ClearAllPoints = function(self)
+		return self.frame:ClearAllPoints()
+	end
+
+	WidgetBase.GetNumPoints = function(self)
+		return self.frame:GetNumPoints()
+	end
+
+	WidgetBase.GetPoint = function(self, ...)
+		return self.frame:GetPoint(...)
+	end
+
+	WidgetBase.GetUserDataTable = function(self)
+		return self.userdata
+	end
+
+	WidgetBase.SetUserData = function(self, key, value)
+		self.userdata[key] = value
+	end
+
+	WidgetBase.GetUserData = function(self, key)
+		return self.userdata[key]
+	end
+
+	WidgetBase.IsFullHeight = function(self)
+		return self.height == "fill"
+	end
+
+	WidgetBase.SetFullHeight = function(self, isFull)
+		if isFull then
+			self.height = "fill"
+		else
+			self.height = nil
+		end
+	end
+
+	WidgetBase.IsFullWidth = function(self)
+		return self.width == "fill"
+	end
+
+	WidgetBase.SetFullWidth = function(self, isFull)
+		if isFull then
+			self.width = "fill"
+		else
+			self.width = nil
+		end
+	end
+
+--	local function LayoutOnUpdate(this)
+--		this:SetScript("OnUpdate",nil)
+--		this.obj:PerformLayout()
+--	end
+
+	local WidgetContainerBase = AceGUI.WidgetContainerBase
+
+	WidgetContainerBase.PauseLayout = function(self)
+		self.LayoutPaused = true
+	end
+
+	WidgetContainerBase.ResumeLayout = function(self)
+		self.LayoutPaused = nil
+	end
+
+	WidgetContainerBase.PerformLayout = function(self)
+		if self.LayoutPaused then
+			return
+		end
+		safecall(self.LayoutFunc, self.content, self.children)
+	end
+
+	--call this function to layout, makes sure layed out objects get a frame to get sizes etc
+	WidgetContainerBase.DoLayout = function(self)
+		self:PerformLayout()
+--		if not self.parent then
+--			self.frame:SetScript("OnUpdate", LayoutOnUpdate)
+--		end
+	end
+
+	WidgetContainerBase.AddChild = function(self, child, beforeWidget)
+		if beforeWidget then
+			local siblingIndex = 1
+			for _, widget in pairs(self.children) do
+				if widget == beforeWidget then
+					break
+				end
+				siblingIndex = siblingIndex + 1
+			end
+			tinsert(self.children, siblingIndex, child)
+		else
+			tinsert(self.children, child)
+		end
+		child:SetParent(self)
+		child.frame:Show()
+		self:DoLayout()
+	end
+
+	WidgetContainerBase.AddChildren = function(self, ...)
+		for i = 1, select("#", ...) do
+			local child = select(i, ...)
+			tinsert(self.children, child)
+			child:SetParent(self)
+			child.frame:Show()
+		end
+		self:DoLayout()
+	end
+
+	WidgetContainerBase.ReleaseChildren = function(self)
+		local children = self.children
+		for i = 1,#children do
+			AceGUI:Release(children[i])
+			children[i] = nil
+		end
+	end
+
+	WidgetContainerBase.SetLayout = function(self, Layout)
+		self.LayoutFunc = AceGUI:GetLayout(Layout)
+	end
+
+	WidgetContainerBase.SetAutoAdjustHeight = function(self, adjust)
+		if adjust then
+			self.noAutoHeight = nil
+		else
+			self.noAutoHeight = true
+		end
+	end
+
+	local function FrameResize(this)
+		local self = this.obj
+		if this:GetWidth() and this:GetHeight() then
+			if self.OnWidthSet then
+				self:OnWidthSet(this:GetWidth())
+			end
+			if self.OnHeightSet then
+				self:OnHeightSet(this:GetHeight())
+			end
+		end
+	end
+
+	local function ContentResize(this)
+		if this:GetWidth() and this:GetHeight() then
+			this.width = this:GetWidth()
+			this.height = this:GetHeight()
+			this.obj:DoLayout()
+		end
+	end
+
+	setmetatable(WidgetContainerBase, {__index=WidgetBase})
+
+	--One of these function should be called on each Widget Instance as part of its creation process
+
+	--- Register a widget-class as a container for newly created widgets.
+	-- @param widget The widget class
+	function AceGUI:RegisterAsContainer(widget)
+		widget.children = {}
+		widget.userdata = {}
+		widget.events = {}
+		widget.base = WidgetContainerBase
+		widget.content.obj = widget
+		widget.frame.obj = widget
+		widget.content:SetScript("OnSizeChanged", ContentResize)
+		widget.frame:SetScript("OnSizeChanged", FrameResize)
+		setmetatable(widget, {__index = WidgetContainerBase})
+		widget:SetLayout("List")
+		return widget
+	end
+
+	--- Register a widget-class as a widget.
+	-- @param widget The widget class
+	function AceGUI:RegisterAsWidget(widget)
+		widget.userdata = {}
+		widget.events = {}
+		widget.base = WidgetBase
+		widget.frame.obj = widget
+		widget.frame:SetScript("OnSizeChanged", FrameResize)
+		setmetatable(widget, {__index = WidgetBase})
+		return widget
+	end
+end
+
+
+
+
+------------------
+-- Widget API   --
+------------------
+
+--- Registers a widget Constructor, this function returns a new instance of the Widget
+-- @param Name The name of the widget
+-- @param Constructor The widget constructor function
+-- @param Version The version of the widget
+function AceGUI:RegisterWidgetType(Name, Constructor, Version)
+	assert(type(Constructor) == "function")
+	assert(type(Version) == "number")
+
+	local oldVersion = WidgetVersions[Name]
+	if oldVersion and oldVersion >= Version then return end
+
+	WidgetVersions[Name] = Version
+	WidgetRegistry[Name] = Constructor
+end
+
+--- Registers a Layout Function
+-- @param Name The name of the layout
+-- @param LayoutFunc Reference to the layout function
+function AceGUI:RegisterLayout(Name, LayoutFunc)
+	assert(type(LayoutFunc) == "function")
+	if type(Name) == "string" then
+		Name = Name:upper()
+	end
+	LayoutRegistry[Name] = LayoutFunc
+end
+
+--- Get a Layout Function from the registry
+-- @param Name The name of the layout
+function AceGUI:GetLayout(Name)
+	if type(Name) == "string" then
+		Name = Name:upper()
+	end
+	return LayoutRegistry[Name]
+end
+
+AceGUI.counts = AceGUI.counts or {}
+
+--- A type-based counter to count the number of widgets created.
+-- This is used by widgets that require a named frame, e.g. when a Blizzard
+-- Template requires it.
+-- @param type The widget type
+function AceGUI:GetNextWidgetNum(type)
+	if not self.counts[type] then
+		self.counts[type] = 0
+	end
+	self.counts[type] = self.counts[type] + 1
+	return self.counts[type]
+end
+
+--- Return the number of created widgets for this type.
+-- In contrast to GetNextWidgetNum, the number is not incremented.
+-- @param type The widget type
+function AceGUI:GetWidgetCount(type)
+	return self.counts[type] or 0
+end
+
+--- Return the version of the currently registered widget type.
+-- @param type The widget type
+function AceGUI:GetWidgetVersion(type)
+	return WidgetVersions[type]
+end
+
+-------------
+-- Layouts --
+-------------
+
+--[[
+	A Layout is a func that takes 2 parameters
+		content - the frame that widgets will be placed inside
+		children - a table containing the widgets to layout
+]]
+
+-- Very simple Layout, Children are stacked on top of each other down the left side
+AceGUI:RegisterLayout("List",
+	function(content, children)
+		local height = 0
+		local width = content.width or content:GetWidth() or 0
+		for i = 1, #children do
+			local child = children[i]
+
+			local frame = child.frame
+			frame:ClearAllPoints()
+			frame:Show()
+			if i == 1 then
+				frame:SetPoint("TOPLEFT", content)
+			else
+				frame:SetPoint("TOPLEFT", children[i-1].frame, "BOTTOMLEFT")
+			end
+
+			if child.width == "fill" then
+				child:SetWidth(width)
+				frame:SetPoint("RIGHT", content)
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+			elseif child.width == "relative" then
+				child:SetWidth(width * child.relWidth)
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+			end
+
+			height = height + (frame.height or frame:GetHeight() or 0)
+		end
+		safecall(content.obj.LayoutFinished, content.obj, nil, height)
+	end)
+
+-- A single control fills the whole content area
+AceGUI:RegisterLayout("Fill",
+	function(content, children)
+		if children[1] then
+			children[1]:SetWidth(content:GetWidth() or 0)
+			children[1]:SetHeight(content:GetHeight() or 0)
+			children[1].frame:SetAllPoints(content)
+			children[1].frame:Show()
+			safecall(content.obj.LayoutFinished, content.obj, nil, children[1].frame:GetHeight())
+		end
+	end)
+
+local layoutrecursionblock = nil
+local function safelayoutcall(object, func, ...)
+	layoutrecursionblock = true
+	object[func](object, ...)
+	layoutrecursionblock = nil
+end
+
+AceGUI:RegisterLayout("Flow",
+	function(content, children)
+		if layoutrecursionblock then return end
+		--used height so far
+		local height = 0
+		--width used in the current row
+		local usedwidth = 0
+		--height of the current row
+		local rowheight = 0
+		local rowoffset = 0
+		local lastrowoffset
+
+		local width = content.width or content:GetWidth() or 0
+
+		--control at the start of the row
+		local rowstart
+		local rowstartoffset
+		local lastrowstart
+		local isfullheight
+
+		local frameoffset
+		local lastframeoffset
+		local oversize
+		for i = 1, #children do
+			local child = children[i]
+			oversize = nil
+			local frame = child.frame
+			local frameheight = frame.height or frame:GetHeight() or 0
+			local framewidth = frame.width or frame:GetWidth() or 0
+			lastframeoffset = frameoffset
+			-- HACK: Why did we set a frameoffset of (frameheight / 2) ?
+			-- That was moving all widgets half the widgets size down, is that intended?
+			-- Actually, it seems to be neccessary for many cases, we'll leave it in for now.
+			-- If widgets seem to anchor weirdly with this, provide a valid alignoffset for them.
+			-- TODO: Investigate moar!
+			frameoffset = child.alignoffset or (frameheight / 2)
+
+			if child.width == "relative" then
+				framewidth = width * child.relWidth
+			end
+
+			frame:Show()
+			frame:ClearAllPoints()
+			if i == 1 then
+				-- anchor the first control to the top left
+				frame:SetPoint("TOPLEFT", content)
+				rowheight = frameheight
+				rowoffset = frameoffset
+				rowstart = frame
+				rowstartoffset = frameoffset
+				usedwidth = framewidth
+				if usedwidth > width then
+					oversize = true
+				end
+			else
+				-- if there isn't available width for the control start a new row
+				-- if a control is "fill" it will be on a row of its own full width
+				if usedwidth == 0 or ((framewidth) + usedwidth > width) or child.width == "fill" then
+					if isfullheight then
+						-- a previous row has already filled the entire height, there's nothing we can usefully do anymore
+						-- (maybe error/warn about this?)
+						break
+					end
+					--anchor the previous row, we will now know its height and offset
+					rowstart:SetPoint("TOPLEFT", content, "TOPLEFT", 0, -(height + (rowoffset - rowstartoffset) + 3))
+					height = height + rowheight + 3
+					--save this as the rowstart so we can anchor it after the row is complete and we have the max height and offset of controls in it
+					rowstart = frame
+					rowstartoffset = frameoffset
+					rowheight = frameheight
+					rowoffset = frameoffset
+					usedwidth = framewidth
+					if usedwidth > width then
+						oversize = true
+					end
+				-- put the control on the current row, adding it to the width and checking if the height needs to be increased
+				else
+					--handles cases where the new height is higher than either control because of the offsets
+					--math.max(rowheight-rowoffset+frameoffset, frameheight-frameoffset+rowoffset)
+
+					--offset is always the larger of the two offsets
+					rowoffset = math_max(rowoffset, frameoffset)
+					rowheight = math_max(rowheight, rowoffset + (frameheight / 2))
+
+					frame:SetPoint("TOPLEFT", children[i-1].frame, "TOPRIGHT", 0, frameoffset - lastframeoffset)
+					usedwidth = framewidth + usedwidth
+				end
+			end
+
+			if child.width == "fill" then
+				safelayoutcall(child, "SetWidth", width)
+				frame:SetPoint("RIGHT", content)
+
+				usedwidth = 0
+				rowstart = frame
+				rowstartoffset = frameoffset
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+				rowheight = frame.height or frame:GetHeight() or 0
+				rowoffset = child.alignoffset or (rowheight / 2)
+				rowstartoffset = rowoffset
+			elseif child.width == "relative" then
+				safelayoutcall(child, "SetWidth", width * child.relWidth)
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+			elseif oversize then
+				if width > 1 then
+					frame:SetPoint("RIGHT", content)
+				end
+			end
+
+			if child.height == "fill" then
+				frame:SetPoint("BOTTOM", content)
+				isfullheight = true
+			end
+		end
+
+		--anchor the last row, if its full height needs a special case since  its height has just been changed by the anchor
+		if isfullheight then
+			rowstart:SetPoint("TOPLEFT", content, "TOPLEFT", 0, -height)
+		elseif rowstart then
+			rowstart:SetPoint("TOPLEFT", content, "TOPLEFT", 0, -(height + (rowoffset - rowstartoffset) + 3))
+		end
+
+		height = height + rowheight + 3
+		safecall(content.obj.LayoutFinished, content.obj, nil, height)
+	end)
+
+-- Get alignment method and value. Possible alignment methods are a callback, a number, "start", "middle", "end", "fill" or "TOPLEFT", "BOTTOMRIGHT" etc.
+local GetCellAlign = function (dir, tableObj, colObj, cellObj, cell, child)
+	local fn = cellObj and (cellObj["align" .. dir] or cellObj.align)
+			or colObj and (colObj["align" .. dir] or colObj.align)
+			or tableObj["align" .. dir] or tableObj.align
+			or "CENTERLEFT"
+	local child, cell, val = child or 0, cell or 0, nil
+
+	if type(fn) == "string" then
+		fn = fn:lower()
+		fn = dir == "V" and (fn:sub(1, 3) == "top" and "start" or fn:sub(1, 6) == "bottom" and "end" or fn:sub(1, 6) == "center" and "middle")
+		  or dir == "H" and (fn:sub(-4) == "left" and "start" or fn:sub(-5) == "right" and "end" or fn:sub(-6) == "center" and "middle")
+		  or fn
+		val = (fn == "start" or fn == "fill") and 0 or fn == "end" and cell - child or (cell - child) / 2
+	elseif type(fn) == "function" then
+		val = fn(child or 0, cell, dir)
+	else
+		val = fn
+	end
+
+	return fn, max(0, min(val, cell))
+end
+
+-- Get width or height for multiple cells combined
+local GetCellDimension = function (dir, laneDim, from, to, space)
+	local dim = 0
+	for cell=from,to do
+		dim = dim + (laneDim[cell] or 0)
+	end
+	return dim + max(0, to - from) * (space or 0)
+end
+
+--[[ Options
+============
+Container:
+ - columns ({col, col, ...}): Column settings. "col" can be a number (<= 0: content width, <1: rel. width, <10: weight, >=10: abs. width) or a table with column setting.
+ - space, spaceH, spaceV: Overall, horizontal and vertical spacing between cells.
+ - align, alignH, alignV: Overall, horizontal and vertical cell alignment. See GetCellAlign() for possible values.
+Columns:
+ - width: Fixed column width (nil or <=0: content width, <1: rel. width, >=1: abs. width).
+ - min or 1: Min width for content based width
+ - max or 2: Max width for content based width
+ - weight: Flexible column width. The leftover width after accounting for fixed-width columns is distributed to weighted columns according to their weights.
+ - align, alignH, alignV: Overwrites the container setting for alignment.
+Cell:
+ - colspan: Makes a cell span multiple columns.
+ - rowspan: Makes a cell span multiple rows.
+ - align, alignH, alignV: Overwrites the container and column setting for alignment.
+]]
+AceGUI:RegisterLayout("Table",
+	function (content, children)
+		local obj = content.obj
+		obj:PauseLayout()
+
+		local tableObj = obj:GetUserData("table")
+		local cols = tableObj.columns
+		local spaceH = tableObj.spaceH or tableObj.space or 0
+		local spaceV = tableObj.spaceV or tableObj.space or 0
+		local totalH = (content:GetWidth() or content.width or 0) - spaceH * (#cols - 1)
+
+		-- We need to reuse these because layout events can come in very frequently
+		local layoutCache = obj:GetUserData("layoutCache")
+		if not layoutCache then
+			layoutCache = {{}, {}, {}, {}, {}, {}}
+			obj:SetUserData("layoutCache", layoutCache)
+		end
+		local t, laneH, laneV, rowspans, rowStart, colStart = unpack(layoutCache)
+
+		-- Create the grid
+		local n, slotFound = 0
+		for i,child in ipairs(children) do
+			if child:IsShown() then
+				repeat
+					n = n + 1
+					local col = (n - 1) % #cols + 1
+					local row = ceil(n / #cols)
+					local rowspan = rowspans[col]
+					local cell = rowspan and rowspan.child or child
+					local cellObj = cell:GetUserData("cell")
+					slotFound = not rowspan
+
+					-- Rowspan
+					if not rowspan and cellObj and cellObj.rowspan then
+						rowspan = {child = child, from = row, to = row + cellObj.rowspan - 1}
+						rowspans[col] = rowspan
+					end
+					if rowspan and i == #children then
+						rowspan.to = row
+					end
+
+					-- Colspan
+					local colspan = max(0, min((cellObj and cellObj.colspan or 1) - 1, #cols - col))
+					n = n + colspan
+
+					-- Place the cell
+					if not rowspan or rowspan.to == row then
+						t[n] = cell
+						rowStart[cell] = rowspan and rowspan.from or row
+						colStart[cell] = col
+
+						if rowspan then
+							rowspans[col] = nil
+						end
+					end
+				until slotFound
+			end
+		end
+
+		local rows = ceil(n / #cols)
+
+		-- Determine fixed size cols and collect weights
+		local extantH, totalWeight = totalH, 0
+		for col,colObj in ipairs(cols) do
+			laneH[col] = 0
+
+			if type(colObj) == "number" then
+				colObj = {[colObj >= 1 and colObj < 10 and "weight" or "width"] = colObj}
+				cols[col] = colObj
+			end
+
+			if colObj.weight then
+				-- Weight
+				totalWeight = totalWeight + (colObj.weight or 1)
+			else
+				if not colObj.width or colObj.width <= 0 then
+					-- Content width
+					for row=1,rows do
+						local child = t[(row - 1) * #cols + col]
+						if child then
+							local f = child.frame
+							f:ClearAllPoints()
+							local childH = f:GetWidth() or 0
+
+							laneH[col] = max(laneH[col], childH - GetCellDimension("H", laneH, colStart[child], col - 1, spaceH))
+						end
+					end
+
+					laneH[col] = max(colObj.min or colObj[1] or 0, min(laneH[col], colObj.max or colObj[2] or laneH[col]))
+				else
+					-- Rel./Abs. width
+					laneH[col] = colObj.width < 1 and colObj.width * totalH or colObj.width
+				end
+				extantH = max(0, extantH - laneH[col])
+			end
+		end
+
+		-- Determine sizes based on weight
+		local scale = totalWeight > 0 and extantH / totalWeight or 0
+		for col,colObj in pairs(cols) do
+			if colObj.weight then
+				laneH[col] = scale * colObj.weight
+			end
+		end
+
+		-- Arrange children
+		for row=1,rows do
+			local rowV = 0
+
+			-- Horizontal placement and sizing
+			for col=1,#cols do
+				local child = t[(row - 1) * #cols + col]
+				if child then
+					local colObj = cols[colStart[child]]
+					local cellObj = child:GetUserData("cell")
+					local offsetH = GetCellDimension("H", laneH, 1, colStart[child] - 1, spaceH) + (colStart[child] == 1 and 0 or spaceH)
+					local cellH = GetCellDimension("H", laneH, colStart[child], col, spaceH)
+
+					local f = child.frame
+					f:ClearAllPoints()
+					local childH = f:GetWidth() or 0
+
+					local alignFn, align = GetCellAlign("H", tableObj, colObj, cellObj, cellH, childH)
+					f:SetPoint("LEFT", content, offsetH + align, 0)
+					if child:IsFullWidth() or alignFn == "fill" or childH > cellH then
+						f:SetPoint("RIGHT", content, "LEFT", offsetH + align + cellH, 0)
+					end
+
+					if child.DoLayout then
+						child:DoLayout()
+					end
+
+					rowV = max(rowV, (f:GetHeight() or 0) - GetCellDimension("V", laneV, rowStart[child], row - 1, spaceV))
+				end
+			end
+
+			laneV[row] = rowV
+
+			-- Vertical placement and sizing
+			for col=1,#cols do
+				local child = t[(row - 1) * #cols + col]
+				if child then
+					local colObj = cols[colStart[child]]
+					local cellObj = child:GetUserData("cell")
+					local offsetV = GetCellDimension("V", laneV, 1, rowStart[child] - 1, spaceV) + (rowStart[child] == 1 and 0 or spaceV)
+					local cellV = GetCellDimension("V", laneV, rowStart[child], row, spaceV)
+
+					local f = child.frame
+					local childV = f:GetHeight() or 0
+
+					local alignFn, align = GetCellAlign("V", tableObj, colObj, cellObj, cellV, childV)
+					if child:IsFullHeight() or alignFn == "fill" then
+						f:SetHeight(cellV)
+					end
+					f:SetPoint("TOP", content, 0, -(offsetV + align))
+				end
+			end
+		end
+
+		-- Calculate total height
+		local totalV = GetCellDimension("V", laneV, 1, #laneV, spaceV)
+
+		-- Cleanup
+		for _,v in pairs(layoutCache) do wipe(v) end
+
+		safecall(obj.LayoutFinished, obj, nil, totalV)
+		obj:ResumeLayout()
+	end)
diff --git a/Titan/libs/AceGUI-3.0/AceGUI-3.0.xml b/Titan/libs/AceGUI-3.0/AceGUI-3.0.xml
new file mode 100755
index 0000000..ae22c90
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/AceGUI-3.0.xml
@@ -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>
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua
new file mode 100755
index 0000000..200fc71
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua
new file mode 100755
index 0000000..18e49cb
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua
new file mode 100755
index 0000000..d1eb471
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua
@@ -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("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("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("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("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("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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua
new file mode 100755
index 0000000..1a330ef
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua
new file mode 100755
index 0000000..81e6e41
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua
@@ -0,0 +1,204 @@
+--[[-----------------------------------------------------------------------------
+ScrollFrame Container
+Plain container that scrolls its content and doesn't grow in height.
+-------------------------------------------------------------------------------]]
+local Type, Version = "ScrollFrame", 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, assert, type = pairs, assert, type
+local min, max, floor, abs = math.min, math.max, math.floor, math.abs
+
+-- 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 = 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
+		local curvalue = self.scrollbar:GetValue()
+		-- 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")
+				self:DoLayout()
+			end
+		else
+			if not self.scrollBarShown then
+				self.scrollBarShown = true
+				self.scrollbar:Show()
+				self.scrollframe:SetPoint("BOTTOMRIGHT", -20, 0)
+				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)
+		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
+	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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua
new file mode 100755
index 0000000..6e23abc
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua
new file mode 100755
index 0000000..1d5b71a
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua
@@ -0,0 +1,350 @@
+--[[-----------------------------------------------------------------------------
+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 status = self.status or self.localstatus
+		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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua
new file mode 100755
index 0000000..ca27bfb
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua
@@ -0,0 +1,720 @@
+--[[-----------------------------------------------------------------------------
+TreeGroup Container
+Container that uses a tree control to switch between groups.
+-------------------------------------------------------------------------------]]
+local Type, Version = "TreeGroup", 41
+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: GameTooltip, 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 frame = self.frame
+	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
+	local normalTexture = button:GetNormalTexture()
+	local line = button.line
+	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("Interface\\Buttons\\UI-PlusButton-UP")
+			toggle:SetPushedTexture("Interface\\Buttons\\UI-PlusButton-DOWN")
+		else
+			toggle:SetNormalTexture("Interface\\Buttons\\UI-MinusButton-UP")
+			toggle:SetPushedTexture("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
+	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
+		GameTooltip:SetOwner(frame, "ANCHOR_NONE")
+		GameTooltip:SetPoint("LEFT",frame,"RIGHT")
+		GameTooltip:SetText(frame.text:GetText() or "", 1, .82, 0, true)
+
+		GameTooltip:Show()
+	end
+end
+
+local function Button_OnLeave(frame)
+	local self = frame.obj
+	self:Fire("OnButtonLeave", frame.uniquevalue, frame)
+
+	if self.enabletooltips then
+		GameTooltip: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 frame = 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:SetPoint("TOPLEFT", frame, "TOPLEFT",0,0)
+	treeframe:SetPoint("BOTTOMLEFT", frame, "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
+		local hasChildren = self.hasChildren
+
+		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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua
new file mode 100755
index 0000000..f572804
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua
@@ -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([[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([[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([[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([[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([[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([[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([[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([[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([[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([[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("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("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
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua
new file mode 100755
index 0000000..dfa76a5
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua
new file mode 100755
index 0000000..345a0e4
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua
@@ -0,0 +1,296 @@
+--[[-----------------------------------------------------------------------------
+Checkbox Widget
+-------------------------------------------------------------------------------]]
+local Type, Version = "CheckBox", 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 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.checkbg, "RIGHT", self.image:GetWidth() + 2, 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:GetHeight())
+			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(self.check, false)
+			self.check:Show()
+		else
+			--Nil is the unknown tristate value
+			if self.tristate and value == nil then
+				SetDesaturation(self.check, true)
+				self.check:Show()
+			else
+				SetDesaturation(self.check, false)
+				self.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("Interface\\Buttons\\UI-RadioButton")
+			checkbg:SetTexCoord(0, 0.25, 0, 1)
+			check:SetTexture("Interface\\Buttons\\UI-RadioButton")
+			check:SetTexCoord(0.25, 0.5, 0, 1)
+			check:SetBlendMode("ADD")
+			highlight:SetTexture("Interface\\Buttons\\UI-RadioButton")
+			highlight:SetTexCoord(0.5, 0.75, 0, 1)
+		else
+			size = 24
+			checkbg:SetTexture("Interface\\Buttons\\UI-CheckBox-Up")
+			checkbg:SetTexCoord(0, 1, 0, 1)
+			check:SetTexture("Interface\\Buttons\\UI-CheckBox-Check")
+			check:SetTexCoord(0, 1, 0, 1)
+			check:SetBlendMode("BLEND")
+			highlight:SetTexture("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:GetHeight())
+		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("Interface\\Buttons\\UI-CheckBox-Up")
+
+	local check = frame:CreateTexture(nil, "OVERLAY")
+	check:SetAllPoints(checkbg)
+	check:SetTexture("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("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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua
new file mode 100755
index 0000000..d0c10b1
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua
@@ -0,0 +1,188 @@
+--[[-----------------------------------------------------------------------------
+ColorPicker Widget
+-------------------------------------------------------------------------------]]
+local Type, Version = "ColorPicker", 23
+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: ShowUIPanel, HideUIPanel, 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)
+	HideUIPanel(ColorPickerFrame)
+	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
+
+		ShowUIPanel(ColorPickerFrame)
+	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("Interface\\ChatFrame\\ChatFrameColorSwatch")
+	colorSwatch:SetPoint("LEFT")
+
+	local texture = frame:CreateTexture(nil, "BACKGROUND")
+	texture:SetWidth(16)
+	texture:SetHeight(16)
+	texture:SetColorTexture(1, 1, 1)
+	texture:SetPoint("CENTER", colorSwatch)
+	texture:Show()
+
+	local checkers = frame:CreateTexture(nil, "BACKGROUND")
+	checkers:SetWidth(14)
+	checkers:SetHeight(14)
+	checkers:SetTexture("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("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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua
new file mode 100755
index 0000000..b13688c
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua
@@ -0,0 +1,471 @@
+--[[ $Id: AceGUIWidget-DropDown-Items.lua 1167 2017-08-29 22:08:48Z funkydude $ ]]--
+
+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("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("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("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
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua
new file mode 100755
index 0000000..93c93df
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua
@@ -0,0 +1,737 @@
+--[[ $Id: AceGUIWidget-DropDown.lua 1167 2017-08-29 22:08:48Z funkydude $ ]]--
+local AceGUI = LibStub("AceGUI-3.0")
+
+-- Lua APIs
+local min, max, floor = math.min, math.max, math.floor
+local select, pairs, ipairs, type = select, pairs, ipairs, type
+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 = 31
+
+	--[[ 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 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)
+
+			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
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua
new file mode 100755
index 0000000..cc1b6a3
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua
new file mode 100755
index 0000000..bf8b2c0
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua
@@ -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("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("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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua
new file mode 100755
index 0000000..fdba650
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua
@@ -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("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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua
new file mode 100755
index 0000000..5ab7fea
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua
@@ -0,0 +1,101 @@
+--[[-----------------------------------------------------------------------------
+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
+
+-- 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
+
+--[[-----------------------------------------------------------------------------
+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)
+
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
new file mode 100755
index 0000000..e77023a
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua
new file mode 100755
index 0000000..cc7c8d5
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua
@@ -0,0 +1,173 @@
+--[[-----------------------------------------------------------------------------
+Label Widget
+Displays text and optionally an icon.
+-------------------------------------------------------------------------------]]
+local Type, Version = "Label", 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 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:GetHeight()
+		else
+			-- image on the left
+			image:SetPoint("TOPLEFT")
+			if image:GetHeight() > label:GetHeight() 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:GetHeight())
+		end
+	else
+		-- no image shown
+		label:SetPoint("TOPLEFT")
+		label:SetWidth(width)
+		height = label:GetHeight()
+	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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
new file mode 100755
index 0000000..c9b0d86
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
@@ -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)
diff --git a/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua
new file mode 100755
index 0000000..7466cb1
--- /dev/null
+++ b/Titan/libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua
@@ -0,0 +1,285 @@
+--[[-----------------------------------------------------------------------------
+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)
+	local self = frame.obj
+	if not frame.setup then
+		local newvalue = frame:GetValue()
+		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)
diff --git a/Titan/libs/AceHook-3.0/AceHook-3.0.lua b/Titan/libs/AceHook-3.0/AceHook-3.0.lua
new file mode 100755
index 0000000..ad91467
--- /dev/null
+++ b/Titan/libs/AceHook-3.0/AceHook-3.0.lua
@@ -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 1118 2014-10-12 08:21:54Z 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
diff --git a/Titan/libs/AceHook-3.0/AceHook-3.0.xml b/Titan/libs/AceHook-3.0/AceHook-3.0.xml
new file mode 100755
index 0000000..ea2ae25
--- /dev/null
+++ b/Titan/libs/AceHook-3.0/AceHook-3.0.xml
@@ -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>
diff --git a/Titan/libs/AceLocale-3.0/AceLocale-3.0.lua b/Titan/libs/AceLocale-3.0/AceLocale-3.0.lua
new file mode 100755
index 0000000..2ecc0cb
--- /dev/null
+++ b/Titan/libs/AceLocale-3.0/AceLocale-3.0.lua
@@ -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
diff --git a/Titan/libs/AceLocale-3.0/AceLocale-3.0.xml b/Titan/libs/AceLocale-3.0/AceLocale-3.0.xml
new file mode 100755
index 0000000..a4cf340
--- /dev/null
+++ b/Titan/libs/AceLocale-3.0/AceLocale-3.0.xml
@@ -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>
diff --git a/Titan/libs/AceTimer-3.0/AceTimer-3.0.lua b/Titan/libs/AceTimer-3.0/AceTimer-3.0.lua
new file mode 100755
index 0000000..7d89500
--- /dev/null
+++ b/Titan/libs/AceTimer-3.0/AceTimer-3.0.lua
@@ -0,0 +1,278 @@
+--- **AceTimer-3.0** provides a central facility for registering timers.
+-- AceTimer supports one-shot timers and repeating timers. All timers are stored in an efficient
+-- data structure that allows easy dispatching and fast rescheduling. Timers can be registered
+-- or canceled at any time, even from within a running timer, without conflict or large overhead.\\
+-- AceTimer is currently limited to firing timers at a frequency of 0.01s as this is what the WoW timer API
+-- restricts us to.
+--
+-- All `:Schedule` functions will return a handle to the current timer, which you will need to store if you
+-- need to cancel the timer you just registered.
+--
+-- **AceTimer-3.0** can be embeded into your addon, either explicitly by calling AceTimer: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 AceTimer itself.\\
+-- It is recommended to embed AceTimer, otherwise you'll have to specify a custom `self` on all calls you
+-- make into AceTimer.
+-- @class file
+-- @name AceTimer-3.0
+-- @release $Id: AceTimer-3.0.lua 1170 2018-03-29 17:38:58Z funkydude $
+
+local MAJOR, MINOR = "AceTimer-3.0", 17 -- Bump minor on changes
+local AceTimer, oldminor = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not AceTimer then return end -- No upgrade needed
+AceTimer.activeTimers = AceTimer.activeTimers or {} -- Active timer list
+local activeTimers = AceTimer.activeTimers -- Upvalue our private data
+
+-- Lua APIs
+local type, unpack, next, error, select = type, unpack, next, error, select
+-- WoW APIs
+local GetTime, C_TimerAfter = GetTime, C_Timer.After
+
+local function new(self, loop, func, delay, ...)
+	if delay < 0.01 then
+		delay = 0.01 -- Restrict to the lowest time that the C_Timer API allows us
+	end
+
+	local timer = {
+		object = self,
+		func = func,
+		looping = loop,
+		argsCount = select("#", ...),
+		delay = delay,
+		ends = GetTime() + delay,
+		...
+	}
+
+	activeTimers[timer] = timer
+
+	-- Create new timer closure to wrap the "timer" object
+	timer.callback = function()
+		if not timer.cancelled then
+			if type(timer.func) == "string" then
+				-- We manually set the unpack count to prevent issues with an arg set that contains nil and ends with nil
+				-- e.g. local t = {1, 2, nil, 3, nil} print(#t) will result in 2, instead of 5. This fixes said issue.
+				timer.object[timer.func](timer.object, unpack(timer, 1, timer.argsCount))
+			else
+				timer.func(unpack(timer, 1, timer.argsCount))
+			end
+
+			if timer.looping and not timer.cancelled then
+				-- Compensate delay to get a perfect average delay, even if individual times don't match up perfectly
+				-- due to fps differences
+				local time = GetTime()
+				local delay = timer.delay - (time - timer.ends)
+				-- Ensure the delay doesn't go below the threshold
+				if delay < 0.01 then delay = 0.01 end
+				C_TimerAfter(delay, timer.callback)
+				timer.ends = time + delay
+			else
+				activeTimers[timer.handle or timer] = nil
+			end
+		end
+	end
+
+	C_TimerAfter(delay, timer.callback)
+	return timer
+end
+
+--- Schedule a new one-shot timer.
+-- The timer will fire once in `delay` seconds, unless canceled before.
+-- @param callback Callback function for the timer pulse (funcref or method name).
+-- @param delay Delay for the timer, in seconds.
+-- @param ... An optional, unlimited amount of arguments to pass to the callback function.
+-- @usage
+-- MyAddOn = LibStub("AceAddon-3.0"):NewAddon("MyAddOn", "AceTimer-3.0")
+--
+-- function MyAddOn:OnEnable()
+--   self:ScheduleTimer("TimerFeedback", 5)
+-- end
+--
+-- function MyAddOn:TimerFeedback()
+--   print("5 seconds passed")
+-- end
+function AceTimer:ScheduleTimer(func, delay, ...)
+	if not func or not delay then
+		error(MAJOR..": ScheduleTimer(callback, delay, args...): 'callback' and 'delay' must have set values.", 2)
+	end
+	if type(func) == "string" then
+		if type(self) ~= "table" then
+			error(MAJOR..": ScheduleTimer(callback, delay, args...): 'self' - must be a table.", 2)
+		elseif not self[func] then
+			error(MAJOR..": ScheduleTimer(callback, delay, args...): Tried to register '"..func.."' as the callback, but it doesn't exist in the module.", 2)
+		end
+	end
+	return new(self, nil, func, delay, ...)
+end
+
+--- Schedule a repeating timer.
+-- The timer will fire every `delay` seconds, until canceled.
+-- @param callback Callback function for the timer pulse (funcref or method name).
+-- @param delay Delay for the timer, in seconds.
+-- @param ... An optional, unlimited amount of arguments to pass to the callback function.
+-- @usage
+-- MyAddOn = LibStub("AceAddon-3.0"):NewAddon("MyAddOn", "AceTimer-3.0")
+--
+-- function MyAddOn:OnEnable()
+--   self.timerCount = 0
+--   self.testTimer = self:ScheduleRepeatingTimer("TimerFeedback", 5)
+-- end
+--
+-- function MyAddOn:TimerFeedback()
+--   self.timerCount = self.timerCount + 1
+--   print(("%d seconds passed"):format(5 * self.timerCount))
+--   -- run 30 seconds in total
+--   if self.timerCount == 6 then
+--     self:CancelTimer(self.testTimer)
+--   end
+-- end
+function AceTimer:ScheduleRepeatingTimer(func, delay, ...)
+	if not func or not delay then
+		error(MAJOR..": ScheduleRepeatingTimer(callback, delay, args...): 'callback' and 'delay' must have set values.", 2)
+	end
+	if type(func) == "string" then
+		if type(self) ~= "table" then
+			error(MAJOR..": ScheduleRepeatingTimer(callback, delay, args...): 'self' - must be a table.", 2)
+		elseif not self[func] then
+			error(MAJOR..": ScheduleRepeatingTimer(callback, delay, args...): Tried to register '"..func.."' as the callback, but it doesn't exist in the module.", 2)
+		end
+	end
+	return new(self, true, func, delay, ...)
+end
+
+--- Cancels a timer with the given id, registered by the same addon object as used for `:ScheduleTimer`
+-- Both one-shot and repeating timers can be canceled with this function, as long as the `id` is valid
+-- and the timer has not fired yet or was canceled before.
+-- @param id The id of the timer, as returned by `:ScheduleTimer` or `:ScheduleRepeatingTimer`
+function AceTimer:CancelTimer(id)
+	local timer = activeTimers[id]
+
+	if not timer then
+		return false
+	else
+		timer.cancelled = true
+		activeTimers[id] = nil
+		return true
+	end
+end
+
+--- Cancels all timers registered to the current addon object ('self')
+function AceTimer:CancelAllTimers()
+	for k,v in next, activeTimers do
+		if v.object == self then
+			AceTimer.CancelTimer(self, k)
+		end
+	end
+end
+
+--- Returns the time left for a timer with the given id, registered by the current addon object ('self').
+-- This function will return 0 when the id is invalid.
+-- @param id The id of the timer, as returned by `:ScheduleTimer` or `:ScheduleRepeatingTimer`
+-- @return The time left on the timer.
+function AceTimer:TimeLeft(id)
+	local timer = activeTimers[id]
+	if not timer then
+		return 0
+	else
+		return timer.ends - GetTime()
+	end
+end
+
+
+-- ---------------------------------------------------------------------
+-- Upgrading
+
+-- Upgrade from old hash-bucket based timers to C_Timer.After timers.
+if oldminor and oldminor < 10 then
+	-- disable old timer logic
+	AceTimer.frame:SetScript("OnUpdate", nil)
+	AceTimer.frame:SetScript("OnEvent", nil)
+	AceTimer.frame:UnregisterAllEvents()
+	-- convert timers
+	for object,timers in next, AceTimer.selfs do
+		for handle,timer in next, timers do
+			if type(timer) == "table" and timer.callback then
+				local newTimer
+				if timer.delay then
+					newTimer = AceTimer.ScheduleRepeatingTimer(timer.object, timer.callback, timer.delay, timer.arg)
+				else
+					newTimer = AceTimer.ScheduleTimer(timer.object, timer.callback, timer.when - GetTime(), timer.arg)
+				end
+				-- Use the old handle for old timers
+				activeTimers[newTimer] = nil
+				activeTimers[handle] = newTimer
+				newTimer.handle = handle
+			end
+		end
+	end
+	AceTimer.selfs = nil
+	AceTimer.hash = nil
+	AceTimer.debug = nil
+elseif oldminor and oldminor < 17 then
+	-- Upgrade from old animation based timers to C_Timer.After timers.
+	AceTimer.inactiveTimers = nil
+	AceTimer.frame = nil
+	local oldTimers = AceTimer.activeTimers
+	-- Clear old timer table and update upvalue
+	AceTimer.activeTimers = {}
+	activeTimers = AceTimer.activeTimers
+	for handle, timer in next, oldTimers do
+		local newTimer
+		-- Stop the old timer animation
+		local duration, elapsed = timer:GetDuration(), timer:GetElapsed()
+		timer:GetParent():Stop()
+		if timer.looping then
+			newTimer = AceTimer.ScheduleRepeatingTimer(timer.object, timer.func, duration, unpack(timer.args, 1, timer.argsCount))
+		else
+			newTimer = AceTimer.ScheduleTimer(timer.object, timer.func, duration - elapsed, unpack(timer.args, 1, timer.argsCount))
+		end
+		-- Use the old handle for old timers
+		activeTimers[newTimer] = nil
+		activeTimers[handle] = newTimer
+		newTimer.handle = handle
+	end
+
+	-- Migrate transitional handles
+	if oldminor < 13 and AceTimer.hashCompatTable then
+		for handle, id in next, AceTimer.hashCompatTable do
+			local t = activeTimers[id]
+			if t then
+				activeTimers[id] = nil
+				activeTimers[handle] = t
+				t.handle = handle
+			end
+		end
+		AceTimer.hashCompatTable = nil
+	end
+end
+
+-- ---------------------------------------------------------------------
+-- Embed handling
+
+AceTimer.embeds = AceTimer.embeds or {}
+
+local mixins = {
+	"ScheduleTimer", "ScheduleRepeatingTimer",
+	"CancelTimer", "CancelAllTimers",
+	"TimeLeft"
+}
+
+function AceTimer:Embed(target)
+	AceTimer.embeds[target] = true
+	for _,v in next, mixins do
+		target[v] = AceTimer[v]
+	end
+	return target
+end
+
+-- AceTimer:OnEmbedDisable(target)
+-- target (object) - target object that AceTimer is embedded in.
+--
+-- cancel all timers registered for the object
+function AceTimer:OnEmbedDisable(target)
+	target:CancelAllTimers()
+end
+
+for addon in next, AceTimer.embeds do
+	AceTimer:Embed(addon)
+end
diff --git a/Titan/libs/AceTimer-3.0/AceTimer-3.0.xml b/Titan/libs/AceTimer-3.0/AceTimer-3.0.xml
new file mode 100755
index 0000000..1987d78
--- /dev/null
+++ b/Titan/libs/AceTimer-3.0/AceTimer-3.0.xml
@@ -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="AceTimer-3.0.lua"/>
+</Ui>
diff --git a/Titan/libs/Bindings.xml b/Titan/libs/Bindings.xml
new file mode 100755
index 0000000..2134442
--- /dev/null
+++ b/Titan/libs/Bindings.xml
@@ -0,0 +1,5 @@
+<Bindings>
+	<Binding name="RELOADUI" description="Reloads the UI." header="ACE3" category="ADDONS">
+		ReloadUI()
+	</Binding>
+</Bindings>
diff --git a/Titan/libs/CHANGES.txt b/Titan/libs/CHANGES.txt
new file mode 100755
index 0000000..f709a8c
--- /dev/null
+++ b/Titan/libs/CHANGES.txt
@@ -0,0 +1,8 @@
+------------------------------------------------------------------------
+r102 | funkydude | 2018-07-17 19:25:42 +0000 (Tue, 17 Jul 2018) | 1 line
+Changed paths:
+   M /trunk/LibSharedMedia-3.0.toc
+
+bump version
+------------------------------------------------------------------------
+
diff --git a/Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.lua b/Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.lua
new file mode 100755
index 0000000..e0fe1af
--- /dev/null
+++ b/Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.lua
@@ -0,0 +1,240 @@
+--[[ $Id: CallbackHandler-1.0.lua 14 2010-08-09 00:43:38Z mikk $ ]]
+local MAJOR, MINOR = "CallbackHandler-1.0", 6
+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 CreateDispatcher(argCount)
+	local code = [[
+	local next, xpcall, eh = ...
+
+	local method, ARGS
+	local function call() method(ARGS) end
+
+	local function dispatch(handlers, ...)
+		local index
+		index, method = next(handlers)
+		if not method then return end
+		local OLD_ARGS = ARGS
+		ARGS = ...
+		repeat
+			xpcall(call, eh)
+			index, method = next(handlers, index)
+		until not method
+		ARGS = OLD_ARGS
+	end
+
+	return dispatch
+	]]
+
+	local ARGS, OLD_ARGS = {}, {}
+	for i = 1, argCount do ARGS[i], OLD_ARGS[i] = "arg"..i, "old_arg"..i end
+	code = code:gsub("OLD_ARGS", tconcat(OLD_ARGS, ", ")):gsub("ARGS", tconcat(ARGS, ", "))
+	return assert(loadstring(code, "safecall Dispatcher["..argCount.."]"))(next, xpcall, errorhandler)
+end
+
+local Dispatchers = setmetatable({}, {__index=function(self, argCount)
+	local dispatcher = CreateDispatcher(argCount)
+	rawset(self, argCount, dispatcher)
+	return dispatcher
+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, OnUsed, OnUnused)
+	-- TODO: Remove this after beta has gone out
+	assert(not OnUsed and not OnUnused, "ACE-80: OnUsed/OnUnused are deprecated. Callbacks are now done to registry.OnUsed and registry.OnUnused")
+
+	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
+
+		Dispatchers[select('#', ...) + 1](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.
+
diff --git a/Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.xml b/Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.xml
new file mode 100755
index 0000000..a5b22a7
--- /dev/null
+++ b/Titan/libs/CallbackHandler-1.0/CallbackHandler-1.0.xml
@@ -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>
diff --git a/Titan/libs/Changelog-Ace3-Release-r1151.txt b/Titan/libs/Changelog-Ace3-Release-r1151.txt
new file mode 100644
index 0000000..c78dc9d
--- /dev/null
+++ b/Titan/libs/Changelog-Ace3-Release-r1151.txt
@@ -0,0 +1,92 @@
+------------------------------------------------------------------------
+r1151 | nevcairiel | 2016-07-18 17:03:29 +0000 (Mon, 18 Jul 2016) | 1 line
+Changed paths:
+   A /tags/Release-r1151 (from /trunk:1150)
+
+Tag as Release-r1151
+------------------------------------------------------------------------
+r1150 | nevcairiel | 2016-07-18 17:02:20 +0000 (Mon, 18 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/changelog.txt
+
+Update changelog
+------------------------------------------------------------------------
+r1149 | nevcairiel | 2016-07-18 17:02:10 +0000 (Mon, 18 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/Ace3.toc
+
+Update TOC for 7.0
+------------------------------------------------------------------------
+r1148 | nevcairiel | 2016-07-18 09:13:02 +0000 (Mon, 18 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua
+   M /trunk/AceTab-3.0/AceTab-3.0.lua
+
+Remove remaining compat code for ancient WoW versions
+------------------------------------------------------------------------
+r1147 | nevcairiel | 2016-07-18 09:10:10 +0000 (Mon, 18 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-Button.lua
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
+
+Remove pre-MoP compat code from AceGUI widgets
+------------------------------------------------------------------------
+r1146 | nevcairiel | 2016-07-18 09:07:16 +0000 (Mon, 18 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
+
+Fix global leak
+------------------------------------------------------------------------
+r1145 | nevcairiel | 2016-07-18 08:32:42 +0000 (Mon, 18 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
+
+AceGUI-3.0: Keybinding: Support mousewheel up/down bindings (Ticket #372)
+------------------------------------------------------------------------
+r1144 | nevcairiel | 2016-07-11 09:04:48 +0000 (Mon, 11 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/changelog.txt
+
+Re-format changelog to be consistent with older entries
+------------------------------------------------------------------------
+r1143 | nevcairiel | 2016-07-11 08:52:03 +0000 (Mon, 11 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceConsole-3.0/AceConsole-3.0.lua
+
+AceConsole-3.0: Fix a typo in the LuaDoc function signature
+------------------------------------------------------------------------
+r1142 | nevcairiel | 2016-07-11 08:36:19 +0000 (Mon, 11 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceDB-3.0/AceDB-3.0.lua
+
+AceDB-3.0: Add the "locale" profile key to the LuaDoc (Ticket #344)
+------------------------------------------------------------------------
+r1141 | nevcairiel | 2016-07-11 08:32:23 +0000 (Mon, 11 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua
+   M /trunk/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
+
+AceGUI-3.0: Expose the HighlightText API for EditBox and MultiLineEditBox Widgets (Ticket #378)
+------------------------------------------------------------------------
+r1140 | nevcairiel | 2016-07-03 07:53:29 +0000 (Sun, 03 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceDBOptions-3.0/AceDBOptions-3.0.lua
+
+AceDBOptions-3.0: Update localization from the WoWAce Localization system
+------------------------------------------------------------------------
+r1139 | nevcairiel | 2016-07-03 07:43:51 +0000 (Sun, 03 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceConfig-3.0/AceConfigDialog-3.0/AceConfigDialog-3.0.lua
+   M /trunk/AceConfig-3.0/AceConfigRegistry-3.0/AceConfigRegistry-3.0.lua
+
+AceConfig-3.0: Allow specifying images using the numeric fileID (Ticket #389)
+------------------------------------------------------------------------
+r1138 | nevcairiel | 2016-06-01 10:17:04 +0000 (Wed, 01 Jun 2016) | 3 lines
+Changed paths:
+   M /trunk/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua
+
+AceGUI-3.0: TreeGroup: Refresh the tree in an OnUpdate once after Acquire, not only after creation
+
+The widget creation and acquire can be on different code paths, this ensures the tree height is always known and the tree populated properly.
+------------------------------------------------------------------------
diff --git a/Titan/libs/Changelog-AceGUI-3.0-SharedMediaWidgets-r57-release.txt b/Titan/libs/Changelog-AceGUI-3.0-SharedMediaWidgets-r57-release.txt
new file mode 100644
index 0000000..8cd4f20
--- /dev/null
+++ b/Titan/libs/Changelog-AceGUI-3.0-SharedMediaWidgets-r57-release.txt
@@ -0,0 +1,27 @@
+------------------------------------------------------------------------
+r57 | nevcairiel | 2016-07-20 21:25:42 +0000 (Wed, 20 Jul 2016) | 1 line
+Changed paths:
+   A /tags/r57-release (from /trunk:56)
+
+Tag as r57-release
+------------------------------------------------------------------------
+r56 | nevcairiel | 2016-07-20 21:22:30 +0000 (Wed, 20 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/.pkgmeta
+   M /trunk/AceGUI-3.0-SharedMediaWidgets/prototypes.lua
+   M /trunk/AceGUI-3.0-SharedMediaWidgets.toc
+
+Properly embed LibSharedMedia-3.0 for stand-alone builds
+------------------------------------------------------------------------
+r54 | nevcairiel | 2016-07-19 06:34:58 +0000 (Tue, 19 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/AceGUI-3.0-SharedMediaWidgets.toc
+
+Update TOC to 7.0
+------------------------------------------------------------------------
+r53 | nevcairiel | 2016-06-03 06:51:46 +0000 (Fri, 03 Jun 2016) | 1 line
+Changed paths:
+   M /trunk/AceGUI-3.0-SharedMediaWidgets/prototypes.lua
+
+Remove SetPoint to UIParent, AceGUI-3.0 will properly position the element when its used
+------------------------------------------------------------------------
diff --git a/Titan/libs/LICENSE.txt b/Titan/libs/LICENSE.txt
new file mode 100755
index 0000000..ef0b614
--- /dev/null
+++ b/Titan/libs/LICENSE.txt
@@ -0,0 +1,29 @@
+Copyright (c) 2007, Ace3 Development Team
+
+All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+    * Redistributions of source code must retain the above copyright notice,
+      this list of conditions and the following disclaimer.
+    * Redistributions in binary form must reproduce the above copyright notice,
+      this list of conditions and the following disclaimer in the documentation
+      and/or other materials provided with the distribution.
+    * Redistribution of a stand alone version is strictly prohibited without
+      prior written authorization from the Lead of the Ace3 Development Team.
+    * Neither the name of the Ace3 Development Team nor the names of its contributors
+      may be used to endorse or promote products derived from this software without
+      specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
+CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
\ No newline at end of file
diff --git a/Titan/libs/LibQTip-1.0/CHANGES.txt b/Titan/libs/LibQTip-1.0/CHANGES.txt
new file mode 100755
index 0000000..157ca0d
--- /dev/null
+++ b/Titan/libs/LibQTip-1.0/CHANGES.txt
@@ -0,0 +1,21 @@
+------------------------------------------------------------------------
+r188 | Torhal | 2018-07-21 06:14:41 +0000 (Sat, 21 Jul 2018) | 1 line
+Changed paths:
+   M /trunk/LibQTip-1.0.toc
+
+Updating ToC Interface to 80000
+------------------------------------------------------------------------
+r187 | Torhal | 2018-07-21 06:13:54 +0000 (Sat, 21 Jul 2018) | 1 line
+Changed paths:
+   A /trunk/LibStub
+   A /trunk/LibStub/LibStub.lua
+
+Hard-embed LibStub, to stop the WoW 8.x client from complaining about being unable to load it under a dev environment.
+------------------------------------------------------------------------
+r186 | Torhal | 2018-07-21 06:13:15 +0000 (Sat, 21 Jul 2018) | 1 line
+Changed paths:
+   M /trunk/.pkgmeta
+
+Hard-embed LibStub, to stop the WoW 8.x client from complaining about being unable to load it under a dev environment.
+------------------------------------------------------------------------
+
diff --git a/Titan/libs/LibQTip-1.0/LICENSE.txt b/Titan/libs/LibQTip-1.0/LICENSE.txt
new file mode 100755
index 0000000..54cdb47
--- /dev/null
+++ b/Titan/libs/LibQTip-1.0/LICENSE.txt
@@ -0,0 +1,29 @@
+Copyright (c) 2008, LibQTip Development Team
+
+All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+    * Redistributions of source code must retain the above copyright notice,
+      this list of conditions and the following disclaimer.
+    * Redistributions in binary form must reproduce the above copyright notice,
+      this list of conditions and the following disclaimer in the documentation
+      and/or other materials provided with the distribution.
+    * Redistribution of a stand alone version is strictly prohibited without
+      prior written authorization from the Lead of the LibQTip Development Team.
+    * Neither the name of the LibQTip Development Team nor the names of its contributors
+      may be used to endorse or promote products derived from this software without
+      specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
+CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
diff --git a/Titan/libs/LibQTip-1.0/LibQTip-1.0.lua b/Titan/libs/LibQTip-1.0/LibQTip-1.0.lua
new file mode 100755
index 0000000..6f58d5b
--- /dev/null
+++ b/Titan/libs/LibQTip-1.0/LibQTip-1.0.lua
@@ -0,0 +1,1529 @@
+local MAJOR = "LibQTip-1.0"
+local MINOR = 46 -- Should be manually increased
+local LibStub = _G.LibStub
+
+assert(LibStub, MAJOR .. " requires LibStub")
+
+local lib, oldMinor = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not lib then
+	return
+end -- No upgrade needed
+
+------------------------------------------------------------------------------
+-- Upvalued globals
+------------------------------------------------------------------------------
+local table = _G.table
+local tinsert = table.insert
+local tremove = table.remove
+local wipe = table.wipe
+
+local error = error
+local math = math
+local min, max = math.min, math.max
+local next = next
+local pairs, ipairs = pairs, ipairs
+local select = select
+local setmetatable = setmetatable
+local tonumber, tostring = tonumber, tostring
+local type = type
+
+local CreateFrame = _G.CreateFrame
+local GameTooltip = _G.GameTooltip
+local UIParent = _G.UIParent
+
+local geterrorhandler = _G.geterrorhandler
+
+------------------------------------------------------------------------------
+-- Tables and locals
+------------------------------------------------------------------------------
+lib.frameMetatable = lib.frameMetatable or {__index = CreateFrame("Frame")}
+
+lib.tipPrototype = lib.tipPrototype or setmetatable({}, lib.frameMetatable)
+lib.tipMetatable = lib.tipMetatable or {__index = lib.tipPrototype}
+
+lib.providerPrototype = lib.providerPrototype or {}
+lib.providerMetatable = lib.providerMetatable or {__index = lib.providerPrototype}
+
+lib.cellPrototype = lib.cellPrototype or setmetatable({}, lib.frameMetatable)
+lib.cellMetatable = lib.cellMetatable or {__index = lib.cellPrototype}
+
+lib.activeTooltips = lib.activeTooltips or {}
+
+lib.tooltipHeap = lib.tooltipHeap or {}
+lib.frameHeap = lib.frameHeap or {}
+lib.tableHeap = lib.tableHeap or {}
+
+lib.onReleaseHandlers = lib.onReleaseHandlers or {}
+
+local tipPrototype = lib.tipPrototype
+local tipMetatable = lib.tipMetatable
+
+local providerPrototype = lib.providerPrototype
+local providerMetatable = lib.providerMetatable
+
+local cellPrototype = lib.cellPrototype
+local cellMetatable = lib.cellMetatable
+
+local activeTooltips = lib.activeTooltips
+
+local highlightFrame = CreateFrame("Frame", nil, UIParent)
+highlightFrame:SetFrameStrata("TOOLTIP")
+highlightFrame:Hide()
+
+local DEFAULT_HIGHLIGHT_TEXTURE_PATH = [[Interface\QuestFrame\UI-QuestTitleHighlight]]
+
+local highlightTexture = highlightFrame:CreateTexture(nil, "OVERLAY")
+highlightTexture:SetTexture(DEFAULT_HIGHLIGHT_TEXTURE_PATH)
+highlightTexture:SetBlendMode("ADD")
+highlightTexture:SetAllPoints(highlightFrame)
+
+------------------------------------------------------------------------------
+-- Private methods for Caches and Tooltip
+------------------------------------------------------------------------------
+local AcquireTooltip, ReleaseTooltip
+local AcquireCell, ReleaseCell
+local AcquireTable, ReleaseTable
+
+local InitializeTooltip, SetTooltipSize, ResetTooltipSize, FixCellSizes
+local ClearTooltipScripts
+local SetFrameScript, ClearFrameScripts
+
+------------------------------------------------------------------------------
+-- Cache debugging.
+------------------------------------------------------------------------------
+-- @debug @
+local usedTables, usedFrames, usedTooltips = 0, 0, 0
+--@end-debug@
+
+------------------------------------------------------------------------------
+-- Internal constants to tweak the layout
+------------------------------------------------------------------------------
+local TOOLTIP_PADDING = 10
+local CELL_MARGIN_H = 6
+local CELL_MARGIN_V = 3
+
+------------------------------------------------------------------------------
+-- Public library API
+------------------------------------------------------------------------------
+--- Create or retrieve the tooltip with the given key.
+-- If additional arguments are passed, they are passed to :SetColumnLayout for the acquired tooltip.
+-- @name LibQTip:Acquire(key[, numColumns, column1Justification, column2justification, ...])
+-- @param key string or table - the tooltip key. Any value that can be used as a table key is accepted though you should try to provide unique keys to avoid conflicts.
+-- Numbers and booleans should be avoided and strings should be carefully chosen to avoid namespace clashes - no "MyTooltip" - you have been warned!
+-- @return tooltip Frame object - the acquired tooltip.
+-- @usage Acquire a tooltip with at least 5 columns, justification : left, center, left, left, left
+-- <pre>local tip = LibStub('LibQTip-1.0'):Acquire('MyFooBarTooltip', 5, "LEFT", "CENTER")</pre>
+function lib:Acquire(key, ...)
+	if key == nil then
+		error("attempt to use a nil key", 2)
+	end
+
+	local tooltip = activeTooltips[key]
+
+	if not tooltip then
+		tooltip = AcquireTooltip()
+		InitializeTooltip(tooltip, key)
+		activeTooltips[key] = tooltip
+	end
+
+	if select("#", ...) > 0 then
+		-- Here we catch any error to properly report it for the calling code
+		local ok, msg = pcall(tooltip.SetColumnLayout, tooltip, ...)
+
+		if not ok then
+			error(msg, 2)
+		end
+	end
+
+	return tooltip
+end
+
+function lib:Release(tooltip)
+	local key = tooltip and tooltip.key
+
+	if not key or activeTooltips[key] ~= tooltip then
+		return
+	end
+
+	ReleaseTooltip(tooltip)
+	activeTooltips[key] = nil
+end
+
+function lib:IsAcquired(key)
+	if key == nil then
+		error("attempt to use a nil key", 2)
+	end
+
+	return not (not activeTooltips[key])
+end
+
+function lib:IterateTooltips()
+	return pairs(activeTooltips)
+end
+
+------------------------------------------------------------------------------
+-- Frame cache
+------------------------------------------------------------------------------
+local frameHeap = lib.frameHeap
+
+local function AcquireFrame(parent)
+	local frame = tremove(frameHeap) or CreateFrame("Frame")
+	frame:SetParent(parent)
+	--[===[@debug@
+	usedFrames = usedFrames + 1
+	--@end-debug@]===]
+	return frame
+end
+
+local function ReleaseFrame(frame)
+	frame:Hide()
+	frame:SetParent(nil)
+	frame:ClearAllPoints()
+	frame:SetBackdrop(nil)
+
+	ClearFrameScripts(frame)
+
+	tinsert(frameHeap, frame)
+	--[===[@debug@
+	usedFrames = usedFrames - 1
+	--@end-debug@]===]
+end
+
+------------------------------------------------------------------------------
+-- Dirty layout handler
+------------------------------------------------------------------------------
+lib.layoutCleaner = lib.layoutCleaner or CreateFrame("Frame")
+
+local layoutCleaner = lib.layoutCleaner
+layoutCleaner.registry = layoutCleaner.registry or {}
+
+function layoutCleaner:RegisterForCleanup(tooltip)
+	self.registry[tooltip] = true
+	self:Show()
+end
+
+function layoutCleaner:CleanupLayouts()
+	self:Hide()
+
+	for tooltip in pairs(self.registry) do
+		FixCellSizes(tooltip)
+	end
+
+	wipe(self.registry)
+end
+
+layoutCleaner:SetScript("OnUpdate", layoutCleaner.CleanupLayouts)
+
+------------------------------------------------------------------------------
+-- CellProvider and Cell
+------------------------------------------------------------------------------
+function providerPrototype:AcquireCell()
+	local cell = tremove(self.heap)
+
+	if not cell then
+		cell = setmetatable(CreateFrame("Frame", nil, UIParent), self.cellMetatable)
+
+		if type(cell.InitializeCell) == "function" then
+			cell:InitializeCell()
+		end
+	end
+
+	self.cells[cell] = true
+
+	return cell
+end
+
+function providerPrototype:ReleaseCell(cell)
+	if not self.cells[cell] then
+		return
+	end
+
+	if type(cell.ReleaseCell) == "function" then
+		cell:ReleaseCell()
+	end
+
+	self.cells[cell] = nil
+	tinsert(self.heap, cell)
+end
+
+function providerPrototype:GetCellPrototype()
+	return self.cellPrototype, self.cellMetatable
+end
+
+function providerPrototype:IterateCells()
+	return pairs(self.cells)
+end
+
+function lib:CreateCellProvider(baseProvider)
+	local cellBaseMetatable, cellBasePrototype
+
+	if baseProvider and baseProvider.GetCellPrototype then
+		cellBasePrototype, cellBaseMetatable = baseProvider:GetCellPrototype()
+	else
+		cellBaseMetatable = cellMetatable
+	end
+
+	local newCellPrototype = setmetatable({}, cellBaseMetatable)
+	local newCellProvider = setmetatable({}, providerMetatable)
+
+	newCellProvider.heap = {}
+	newCellProvider.cells = {}
+	newCellProvider.cellPrototype = newCellPrototype
+	newCellProvider.cellMetatable = {__index = newCellPrototype}
+
+	return newCellProvider, newCellPrototype, cellBasePrototype
+end
+
+------------------------------------------------------------------------------
+-- Basic label provider
+------------------------------------------------------------------------------
+if not lib.LabelProvider then
+	lib.LabelProvider, lib.LabelPrototype = lib:CreateCellProvider()
+end
+
+local labelProvider = lib.LabelProvider
+local labelPrototype = lib.LabelPrototype
+
+function labelPrototype:InitializeCell()
+	self.fontString = self:CreateFontString()
+	self.fontString:SetFontObject(_G.GameTooltipText)
+end
+
+function labelPrototype:SetupCell(tooltip, value, justification, font, leftPadding, rightPadding, maxWidth, minWidth, ...)
+	local fontString = self.fontString
+	local line = tooltip.lines[self._line]
+
+	-- detatch fs from cell for size calculations
+	fontString:ClearAllPoints()
+	fontString:SetFontObject(font or (line.is_header and tooltip:GetHeaderFont() or tooltip:GetFont()))
+	fontString:SetJustifyH(justification)
+	fontString:SetText(tostring(value))
+
+	leftPadding = leftPadding or 0
+	rightPadding = rightPadding or 0
+
+	local width = fontString:GetStringWidth() + leftPadding + rightPadding
+
+	if maxWidth and minWidth and (maxWidth < minWidth) then
+		error("maximum width cannot be lower than minimum width: " .. tostring(maxWidth) .. " < " .. tostring(minWidth), 2)
+	end
+
+	if maxWidth and (maxWidth < (leftPadding + rightPadding)) then
+		error("maximum width cannot be lower than the sum of paddings: " .. tostring(maxWidth) .. " < " .. tostring(leftPadding) .. " + " .. tostring(rightPadding), 2)
+	end
+
+	if minWidth and width < minWidth then
+		width = minWidth
+	end
+
+	if maxWidth and maxWidth < width then
+		width = maxWidth
+	end
+
+	fontString:SetWidth(width - (leftPadding + rightPadding))
+	-- Use GetHeight() instead of GetStringHeight() so lines which are longer than width will wrap.
+	local height = fontString:GetHeight()
+
+	-- reanchor fs to cell
+	fontString:SetWidth(0)
+	fontString:SetPoint("TOPLEFT", self, "TOPLEFT", leftPadding, 0)
+	fontString:SetPoint("BOTTOMRIGHT", self, "BOTTOMRIGHT", -rightPadding, 0)
+	--~ 	fs:SetPoint("TOPRIGHT", self, "TOPRIGHT", -r_pad, 0)
+
+	self._paddingL = leftPadding
+	self._paddingR = rightPadding
+
+	return width, height
+end
+
+function labelPrototype:getContentHeight()
+	local fontString = self.fontString
+	fontString:SetWidth(self:GetWidth() - (self._paddingL + self._paddingR))
+
+	local height = self.fontString:GetHeight()
+	fontString:SetWidth(0)
+
+	return height
+end
+
+function labelPrototype:GetPosition()
+	return self._line, self._column
+end
+
+------------------------------------------------------------------------------
+-- Tooltip cache
+------------------------------------------------------------------------------
+local tooltipHeap = lib.tooltipHeap
+
+-- Returns a tooltip
+function AcquireTooltip()
+	local tooltip = tremove(tooltipHeap)
+
+	if not tooltip then
+		tooltip = CreateFrame("Frame", nil, UIParent)
+
+		local scrollFrame = CreateFrame("ScrollFrame", nil, tooltip)
+		scrollFrame:SetPoint("TOP", tooltip, "TOP", 0, -TOOLTIP_PADDING)
+		scrollFrame:SetPoint("BOTTOM", tooltip, "BOTTOM", 0, TOOLTIP_PADDING)
+		scrollFrame:SetPoint("LEFT", tooltip, "LEFT", TOOLTIP_PADDING, 0)
+		scrollFrame:SetPoint("RIGHT", tooltip, "RIGHT", -TOOLTIP_PADDING, 0)
+		tooltip.scrollFrame = scrollFrame
+
+		local scrollChild = CreateFrame("Frame", nil, tooltip.scrollFrame)
+		scrollFrame:SetScrollChild(scrollChild)
+		tooltip.scrollChild = scrollChild
+
+		setmetatable(tooltip, tipMetatable)
+	end
+
+	--[===[@debug@
+	usedTooltips = usedTooltips + 1
+	--@end-debug@]===]
+	return tooltip
+end
+
+-- Cleans the tooltip and stores it in the cache
+function ReleaseTooltip(tooltip)
+	if tooltip.releasing then
+		return
+	end
+
+	tooltip.releasing = true
+	tooltip:Hide()
+
+	local releaseHandler = lib.onReleaseHandlers[tooltip]
+
+	if releaseHandler then
+		lib.onReleaseHandlers[tooltip] = nil
+
+		local success, errorMessage = pcall(releaseHandler, tooltip)
+
+		if not success then
+			geterrorhandler()(errorMessage)
+		end
+	elseif tooltip.OnRelease then
+		local success, errorMessage = pcall(tooltip.OnRelease, tooltip)
+		if not success then
+			geterrorhandler()(errorMessage)
+		end
+
+		tooltip.OnRelease = nil
+	end
+
+	tooltip.releasing = nil
+	tooltip.key = nil
+	tooltip.step = nil
+
+	ClearTooltipScripts(tooltip)
+
+	tooltip:SetAutoHideDelay(nil)
+	tooltip:ClearAllPoints()
+	tooltip:Clear()
+
+	if tooltip.slider then
+		tooltip.slider:SetValue(0)
+		tooltip.slider:Hide()
+		tooltip.scrollFrame:SetPoint("RIGHT", tooltip, "RIGHT", -TOOLTIP_PADDING, 0)
+		tooltip:EnableMouseWheel(false)
+	end
+
+	for i, column in ipairs(tooltip.columns) do
+		tooltip.columns[i] = ReleaseFrame(column)
+	end
+
+	tooltip.columns = ReleaseTable(tooltip.columns)
+	tooltip.lines = ReleaseTable(tooltip.lines)
+	tooltip.colspans = ReleaseTable(tooltip.colspans)
+
+	layoutCleaner.registry[tooltip] = nil
+	tinsert(tooltipHeap, tooltip)
+
+	highlightTexture:SetTexture(DEFAULT_HIGHLIGHT_TEXTURE_PATH)
+	highlightTexture:SetTexCoord(0, 1, 0, 1)
+
+	--[===[@debug@
+	usedTooltips = usedTooltips - 1
+	--@end-debug@]===]
+end
+
+------------------------------------------------------------------------------
+-- Cell 'cache' (just a wrapper to the provider's cache)
+------------------------------------------------------------------------------
+-- Returns a cell for the given tooltip from the given provider
+function AcquireCell(tooltip, provider)
+	local cell = provider:AcquireCell(tooltip)
+
+	cell:SetParent(tooltip.scrollChild)
+	cell:SetFrameLevel(tooltip.scrollChild:GetFrameLevel() + 3)
+	cell._provider = provider
+
+	return cell
+end
+
+-- Cleans the cell hands it to its provider for storing
+function ReleaseCell(cell)
+	if cell.fontString and cell.r then
+		cell.fontString:SetTextColor(cell.r, cell.g, cell.b, cell.a)
+	end
+
+	cell._font = nil
+	cell._justification = nil
+	cell._colSpan = nil
+	cell._line = nil
+	cell._column = nil
+
+	cell:Hide()
+	cell:ClearAllPoints()
+	cell:SetParent(nil)
+	cell:SetBackdrop(nil)
+
+	ClearFrameScripts(cell)
+
+	cell._provider:ReleaseCell(cell)
+	cell._provider = nil
+end
+
+------------------------------------------------------------------------------
+-- Table cache
+------------------------------------------------------------------------------
+local tableHeap = lib.tableHeap
+
+-- Returns a table
+function AcquireTable()
+	local tbl = tremove(tableHeap) or {}
+	--[===[@debug@
+	usedTables = usedTables + 1
+	--@end-debug@]===]
+	return tbl
+end
+
+-- Cleans the table and stores it in the cache
+function ReleaseTable(tableInstance)
+	wipe(tableInstance)
+	tinsert(tableHeap, tableInstance)
+	--[===[@debug@
+	usedTables = usedTables - 1
+	--@end-debug@]===]
+end
+
+------------------------------------------------------------------------------
+-- Tooltip prototype
+------------------------------------------------------------------------------
+function InitializeTooltip(tooltip, key)
+	----------------------------------------------------------------------
+	-- (Re)set frame settings
+	----------------------------------------------------------------------
+	local backdrop = GameTooltip:GetBackdrop()
+
+	tooltip:SetBackdrop(backdrop)
+
+	if backdrop then
+		tooltip:SetBackdropColor(GameTooltip:GetBackdropColor())
+		tooltip:SetBackdropBorderColor(GameTooltip:GetBackdropBorderColor())
+	end
+
+	tooltip:SetScale(GameTooltip:GetScale())
+	tooltip:SetAlpha(1)
+	tooltip:SetFrameStrata("TOOLTIP")
+	tooltip:SetClampedToScreen(false)
+
+	----------------------------------------------------------------------
+	-- Internal data. Since it's possible to Acquire twice without calling
+	-- release, check for pre-existence.
+	----------------------------------------------------------------------
+	tooltip.key = key
+	tooltip.columns = tooltip.columns or AcquireTable()
+	tooltip.lines = tooltip.lines or AcquireTable()
+	tooltip.colspans = tooltip.colspans or AcquireTable()
+	tooltip.regularFont = _G.GameTooltipText
+	tooltip.headerFont = _G.GameTooltipHeaderText
+	tooltip.labelProvider = labelProvider
+	tooltip.cell_margin_h = tooltip.cell_margin_h or CELL_MARGIN_H
+	tooltip.cell_margin_v = tooltip.cell_margin_v or CELL_MARGIN_V
+
+	----------------------------------------------------------------------
+	-- Finishing procedures
+	----------------------------------------------------------------------
+	tooltip:SetAutoHideDelay(nil)
+	tooltip:Hide()
+	ResetTooltipSize(tooltip)
+end
+
+function tipPrototype:SetDefaultProvider(myProvider)
+	if not myProvider then
+		return
+	end
+
+	self.labelProvider = myProvider
+end
+
+function tipPrototype:GetDefaultProvider()
+	return self.labelProvider
+end
+
+local function checkJustification(justification, level, silent)
+	if justification ~= "LEFT" and justification ~= "CENTER" and justification ~= "RIGHT" then
+		if silent then
+			return false
+		end
+		error("invalid justification, must one of LEFT, CENTER or RIGHT, not: " .. tostring(justification), level + 1)
+	end
+
+	return true
+end
+
+function tipPrototype:SetColumnLayout(numColumns, ...)
+	if type(numColumns) ~= "number" or numColumns < 1 then
+		error("number of columns must be a positive number, not: " .. tostring(numColumns), 2)
+	end
+
+	for i = 1, numColumns do
+		local justification = select(i, ...) or "LEFT"
+
+		checkJustification(justification, 2)
+
+		if self.columns[i] then
+			self.columns[i].justification = justification
+		else
+			self:AddColumn(justification)
+		end
+	end
+end
+
+function tipPrototype:AddColumn(justification)
+	justification = justification or "LEFT"
+	checkJustification(justification, 2)
+
+	local colNum = #self.columns + 1
+	local column = self.columns[colNum] or AcquireFrame(self.scrollChild)
+
+	column:SetFrameLevel(self.scrollChild:GetFrameLevel() + 1)
+	column.justification = justification
+	column.width = 0
+	column:SetWidth(1)
+	column:SetPoint("TOP", self.scrollChild)
+	column:SetPoint("BOTTOM", self.scrollChild)
+
+	if colNum > 1 then
+		local h_margin = self.cell_margin_h or CELL_MARGIN_H
+
+		column:SetPoint("LEFT", self.columns[colNum - 1], "RIGHT", h_margin, 0)
+		SetTooltipSize(self, self.width + h_margin, self.height)
+	else
+		column:SetPoint("LEFT", self.scrollChild)
+	end
+
+	column:Show()
+	self.columns[colNum] = column
+
+	return colNum
+end
+
+------------------------------------------------------------------------------
+-- Convenient methods
+------------------------------------------------------------------------------
+function tipPrototype:Release()
+	lib:Release(self)
+end
+
+function tipPrototype:IsAcquiredBy(key)
+	return key ~= nil and self.key == key
+end
+
+------------------------------------------------------------------------------
+-- Script hooks
+------------------------------------------------------------------------------
+local RawSetScript = lib.frameMetatable.__index.SetScript
+
+function ClearTooltipScripts(tooltip)
+	if tooltip.scripts then
+		for scriptType in pairs(tooltip.scripts) do
+			RawSetScript(tooltip, scriptType, nil)
+		end
+
+		tooltip.scripts = ReleaseTable(tooltip.scripts)
+	end
+end
+
+function tipPrototype:SetScript(scriptType, handler)
+	RawSetScript(self, scriptType, handler)
+
+	if handler then
+		if not self.scripts then
+			self.scripts = AcquireTable()
+		end
+
+		self.scripts[scriptType] = true
+	elseif self.scripts then
+		self.scripts[scriptType] = nil
+	end
+end
+
+-- That might break some addons ; those addons were breaking other
+-- addons' tooltip though.
+function tipPrototype:HookScript()
+	geterrorhandler()(":HookScript is not allowed on LibQTip tooltips")
+end
+
+------------------------------------------------------------------------------
+-- Scrollbar data and functions
+------------------------------------------------------------------------------
+local sliderBackdrop = {
+	bgFile = [[Interface\Buttons\UI-SliderBar-Background]],
+	edgeFile = [[Interface\Buttons\UI-SliderBar-Border]],
+	tile = true,
+	edgeSize = 8,
+	tileSize = 8,
+	insets = {
+		left = 3,
+		right = 3,
+		top = 3,
+		bottom = 3
+	}
+}
+
+local function slider_OnValueChanged(self)
+	self.scrollFrame:SetVerticalScroll(self:GetValue())
+end
+
+local function tooltip_OnMouseWheel(self, delta)
+	local slider = self.slider
+	local currentValue = slider:GetValue()
+	local minValue, maxValue = slider:GetMinMaxValues()
+	local stepValue = self.step or 10
+
+	if delta < 0 and currentValue < maxValue then
+		slider:SetValue(min(maxValue, currentValue + stepValue))
+	elseif delta > 0 and currentValue > minValue then
+		slider:SetValue(max(minValue, currentValue - stepValue))
+	end
+end
+
+-- Set the step size for the scroll bar
+function tipPrototype:SetScrollStep(step)
+	self.step = step
+end
+
+-- will resize the tooltip to fit the screen and show a scrollbar if needed
+function tipPrototype:UpdateScrolling(maxheight)
+	self:SetClampedToScreen(false)
+
+	-- all data is in the tooltip; fix colspan width and prevent the layout cleaner from messing up the tooltip later
+	FixCellSizes(self)
+	layoutCleaner.registry[self] = nil
+
+	local scale = self:GetScale()
+	local topside = self:GetTop()
+	local bottomside = self:GetBottom()
+	local screensize = UIParent:GetHeight() / scale
+	local tipsize = (topside - bottomside)
+
+	-- if the tooltip would be too high, limit its height and show the slider
+	if bottomside < 0 or topside > screensize or (maxheight and tipsize > maxheight) then
+		local shrink = (bottomside < 0 and (5 - bottomside) or 0) + (topside > screensize and (topside - screensize + 5) or 0)
+
+		if maxheight and tipsize - shrink > maxheight then
+			shrink = tipsize - maxheight
+		end
+
+		self:SetHeight(2 * TOOLTIP_PADDING + self.height - shrink)
+		self:SetWidth(2 * TOOLTIP_PADDING + self.width + 20)
+		self.scrollFrame:SetPoint("RIGHT", self, "RIGHT", -(TOOLTIP_PADDING + 20), 0)
+
+		if not self.slider then
+			local slider = CreateFrame("Slider", nil, self)
+			slider.scrollFrame = self.scrollFrame
+
+			slider:SetOrientation("VERTICAL")
+			slider:SetPoint("TOPRIGHT", self, "TOPRIGHT", -TOOLTIP_PADDING, -TOOLTIP_PADDING)
+			slider:SetPoint("BOTTOMRIGHT", self, "BOTTOMRIGHT", -TOOLTIP_PADDING, TOOLTIP_PADDING)
+			slider:SetBackdrop(sliderBackdrop)
+			slider:SetThumbTexture([[Interface\Buttons\UI-SliderBar-Button-Vertical]])
+			slider:SetMinMaxValues(0, 1)
+			slider:SetValueStep(1)
+			slider:SetWidth(12)
+			slider:SetScript("OnValueChanged", slider_OnValueChanged)
+			slider:SetValue(0)
+
+			self.slider = slider
+		end
+
+		self.slider:SetMinMaxValues(0, shrink)
+		self.slider:Show()
+
+		self:EnableMouseWheel(true)
+		self:SetScript("OnMouseWheel", tooltip_OnMouseWheel)
+	else
+		self:SetHeight(2 * TOOLTIP_PADDING + self.height)
+		self:SetWidth(2 * TOOLTIP_PADDING + self.width)
+
+		self.scrollFrame:SetPoint("RIGHT", self, "RIGHT", -TOOLTIP_PADDING, 0)
+
+		if self.slider then
+			self.slider:SetValue(0)
+			self.slider:Hide()
+
+			self:EnableMouseWheel(false)
+			self:SetScript("OnMouseWheel", nil)
+		end
+	end
+end
+
+------------------------------------------------------------------------------
+-- Tooltip methods for changing its contents.
+------------------------------------------------------------------------------
+function tipPrototype:Clear()
+	for i, line in ipairs(self.lines) do
+		for _, cell in pairs(line.cells) do
+			if cell then
+				ReleaseCell(cell)
+			end
+		end
+
+		ReleaseTable(line.cells)
+
+		line.cells = nil
+		line.is_header = nil
+
+		ReleaseFrame(line)
+
+		self.lines[i] = nil
+	end
+
+	for _, column in ipairs(self.columns) do
+		column.width = 0
+		column:SetWidth(1)
+	end
+
+	wipe(self.colspans)
+
+	self.cell_margin_h = nil
+	self.cell_margin_v = nil
+
+	ResetTooltipSize(self)
+end
+
+function tipPrototype:SetCellMarginH(size)
+	if #self.lines > 0 then
+		error("Unable to set horizontal margin while the tooltip has lines.", 2)
+	end
+
+	if not size or type(size) ~= "number" or size < 0 then
+		error("Margin size must be a positive number or zero.", 2)
+	end
+
+	self.cell_margin_h = size
+end
+
+function tipPrototype:SetCellMarginV(size)
+	if #self.lines > 0 then
+		error("Unable to set vertical margin while the tooltip has lines.", 2)
+	end
+
+	if not size or type(size) ~= "number" or size < 0 then
+		error("Margin size must be a positive number or zero.", 2)
+	end
+
+	self.cell_margin_v = size
+end
+
+function SetTooltipSize(tooltip, width, height)
+	tooltip.height = height
+	tooltip.width = width
+
+	tooltip:SetHeight(2 * TOOLTIP_PADDING + height)
+	tooltip:SetWidth(2 * TOOLTIP_PADDING + width)
+
+	tooltip.scrollChild:SetHeight(height)
+	tooltip.scrollChild:SetWidth(width)
+end
+
+-- Add 2 pixels to height so dangling letters (g, y, p, j, etc) are not clipped.
+function ResetTooltipSize(tooltip)
+	local h_margin = tooltip.cell_margin_h or CELL_MARGIN_H
+
+	SetTooltipSize(tooltip, max(0, (h_margin * (#tooltip.columns - 1)) + (h_margin / 2)), 2)
+end
+
+local function EnlargeColumn(tooltip, column, width)
+	if width > column.width then
+		SetTooltipSize(tooltip, tooltip.width + width - column.width, tooltip.height)
+
+		column.width = width
+		column:SetWidth(width)
+	end
+end
+
+local function ResizeLine(tooltip, line, height)
+	SetTooltipSize(tooltip, tooltip.width, tooltip.height + height - line.height)
+
+	line.height = height
+	line:SetHeight(height)
+end
+
+function FixCellSizes(tooltip)
+	local columns = tooltip.columns
+	local colspans = tooltip.colspans
+	local lines = tooltip.lines
+	local h_margin = tooltip.cell_margin_h or CELL_MARGIN_H
+
+	-- resize columns to make room for the colspans
+	while next(colspans) do
+		local maxNeedCols
+		local maxNeedWidthPerCol = 0
+
+		-- calculate the colspan with the highest additional width need per column
+		for colRange, width in pairs(colspans) do
+			local left, right = colRange:match("^(%d+)%-(%d+)$")
+
+			left, right = tonumber(left), tonumber(right)
+
+			for col = left, right - 1 do
+				width = width - columns[col].width - h_margin
+			end
+
+			width = width - columns[right].width
+
+			if width <= 0 then
+				colspans[colRange] = nil
+			else
+				width = width / (right - left + 1)
+
+				if width > maxNeedWidthPerCol then
+					maxNeedCols = colRange
+					maxNeedWidthPerCol = width
+				end
+			end
+		end
+
+		-- resize all columns for that colspan
+		if maxNeedCols then
+			local left, right = maxNeedCols:match("^(%d+)%-(%d+)$")
+
+			for col = left, right do
+				EnlargeColumn(tooltip, columns[col], columns[col].width + maxNeedWidthPerCol)
+			end
+
+			colspans[maxNeedCols] = nil
+		end
+	end
+
+	--now that the cell width is set, recalculate the rows' height
+	for _, line in ipairs(lines) do
+		if #(line.cells) > 0 then
+			local lineheight = 0
+
+			for _, cell in pairs(line.cells) do
+				if cell then
+					lineheight = max(lineheight, cell:getContentHeight())
+				end
+			end
+
+			if lineheight > 0 then
+				ResizeLine(tooltip, line, lineheight)
+			end
+		end
+	end
+end
+
+local function _SetCell(tooltip, lineNum, colNum, value, font, justification, colSpan, provider, ...)
+	local line = tooltip.lines[lineNum]
+	local cells = line.cells
+
+	-- Unset: be quick
+	if value == nil then
+		local cell = cells[colNum]
+
+		if cell then
+			for i = colNum, colNum + cell._colSpan - 1 do
+				cells[i] = nil
+			end
+
+			ReleaseCell(cell)
+		end
+
+		return lineNum, colNum
+	end
+
+	font = font or (line.is_header and tooltip.headerFont or tooltip.regularFont)
+
+	-- Check previous cell
+	local cell
+	local prevCell = cells[colNum]
+
+	if prevCell then
+		-- There is a cell here
+		justification = justification or prevCell._justification
+		colSpan = colSpan or prevCell._colSpan
+
+		-- Clear the currently marked colspan
+		for i = colNum + 1, colNum + prevCell._colSpan - 1 do
+			cells[i] = nil
+		end
+
+		if provider == nil or prevCell._provider == provider then
+			-- Reuse existing cell
+			cell = prevCell
+			provider = cell._provider
+		else
+			-- A new cell is required
+			cells[colNum] = ReleaseCell(prevCell)
+		end
+	elseif prevCell == nil then
+		-- Creating a new cell, using meaningful defaults.
+		provider = provider or tooltip.labelProvider
+		justification = justification or tooltip.columns[colNum].justification or "LEFT"
+		colSpan = colSpan or 1
+	else
+		error("overlapping cells at column " .. colNum, 3)
+	end
+
+	local tooltipWidth = #tooltip.columns
+	local rightColNum
+
+	if colSpan > 0 then
+		rightColNum = colNum + colSpan - 1
+
+		if rightColNum > tooltipWidth then
+			error("ColSpan too big, cell extends beyond right-most column", 3)
+		end
+	else
+		-- Zero or negative: count back from right-most columns
+		rightColNum = max(colNum, tooltipWidth + colSpan)
+		-- Update colspan to its effective value
+		colSpan = 1 + rightColNum - colNum
+	end
+
+	-- Cleanup colspans
+	for i = colNum + 1, rightColNum do
+		local columnCell = cells[i]
+
+		if columnCell then
+			ReleaseCell(columnCell)
+		elseif columnCell == false then
+			error("overlapping cells at column " .. i, 3)
+		end
+
+		cells[i] = false
+	end
+
+	-- Create the cell
+	if not cell then
+		cell = AcquireCell(tooltip, provider)
+		cells[colNum] = cell
+	end
+
+	-- Anchor the cell
+	cell:SetPoint("LEFT", tooltip.columns[colNum])
+	cell:SetPoint("RIGHT", tooltip.columns[rightColNum])
+	cell:SetPoint("TOP", line)
+	cell:SetPoint("BOTTOM", line)
+
+	-- Store the cell settings directly into the cell
+	-- That's a bit risky but is really cheap compared to other ways to do it
+	cell._font, cell._justification, cell._colSpan, cell._line, cell._column = font, justification, colSpan, lineNum, colNum
+
+	-- Setup the cell content
+	local width, height = cell:SetupCell(tooltip, value, justification, font, ...)
+	cell:Show()
+
+	if colSpan > 1 then
+		-- Postpone width changes until the tooltip is shown
+		local colRange = colNum .. "-" .. rightColNum
+
+		tooltip.colspans[colRange] = max(tooltip.colspans[colRange] or 0, width)
+		layoutCleaner:RegisterForCleanup(tooltip)
+	else
+		-- Enlarge the column and tooltip if need be
+		EnlargeColumn(tooltip, tooltip.columns[colNum], width)
+	end
+
+	-- Enlarge the line and tooltip if need be
+	if height > line.height then
+		SetTooltipSize(tooltip, tooltip.width, tooltip.height + height - line.height)
+
+		line.height = height
+		line:SetHeight(height)
+	end
+
+	if rightColNum < tooltipWidth then
+		return lineNum, rightColNum + 1
+	else
+		return lineNum, nil
+	end
+end
+
+do
+	local function CreateLine(tooltip, font, ...)
+		if #tooltip.columns == 0 then
+			error("column layout should be defined before adding line", 3)
+		end
+
+		local lineNum = #tooltip.lines + 1
+		local line = tooltip.lines[lineNum] or AcquireFrame(tooltip.scrollChild)
+
+		line:SetFrameLevel(tooltip.scrollChild:GetFrameLevel() + 2)
+		line:SetPoint("LEFT", tooltip.scrollChild)
+		line:SetPoint("RIGHT", tooltip.scrollChild)
+
+		if lineNum > 1 then
+			local v_margin = tooltip.cell_margin_v or CELL_MARGIN_V
+
+			line:SetPoint("TOP", tooltip.lines[lineNum - 1], "BOTTOM", 0, -v_margin)
+			SetTooltipSize(tooltip, tooltip.width, tooltip.height + v_margin)
+		else
+			line:SetPoint("TOP", tooltip.scrollChild)
+		end
+
+		tooltip.lines[lineNum] = line
+
+		line.cells = line.cells or AcquireTable()
+		line.height = 0
+		line:SetHeight(1)
+		line:Show()
+
+		local colNum = 1
+
+		for i = 1, #tooltip.columns do
+			local value = select(i, ...)
+
+			if value ~= nil then
+				lineNum, colNum = _SetCell(tooltip, lineNum, i, value, font, nil, 1, tooltip.labelProvider)
+			end
+		end
+
+		return lineNum, colNum
+	end
+
+	function tipPrototype:AddLine(...)
+		return CreateLine(self, self.regularFont, ...)
+	end
+
+	function tipPrototype:AddHeader(...)
+		local line, col = CreateLine(self, self.headerFont, ...)
+
+		self.lines[line].is_header = true
+
+		return line, col
+	end
+end -- do-block
+
+local GenericBackdrop = {
+	bgFile = "Interface\\Tooltips\\UI-Tooltip-Background"
+}
+
+function tipPrototype:AddSeparator(height, r, g, b, a)
+	local lineNum, colNum = self:AddLine()
+	local line = self.lines[lineNum]
+	local color = _G.NORMAL_FONT_COLOR
+
+	height = height or 1
+
+	SetTooltipSize(self, self.width, self.height + height)
+
+	line.height = height
+	line:SetHeight(height)
+	line:SetBackdrop(GenericBackdrop)
+	line:SetBackdropColor(r or color.r, g or color.g, b or color.b, a or 1)
+
+	return lineNum, colNum
+end
+
+function tipPrototype:SetCellColor(lineNum, colNum, r, g, b, a)
+	local cell = self.lines[lineNum].cells[colNum]
+
+	if cell then
+		local sr, sg, sb, sa = self:GetBackdropColor()
+
+		cell:SetBackdrop(GenericBackdrop)
+		cell:SetBackdropColor(r or sr, g or sg, b or sb, a or sa)
+	end
+end
+
+function tipPrototype:SetColumnColor(colNum, r, g, b, a)
+	local column = self.columns[colNum]
+
+	if column then
+		local sr, sg, sb, sa = self:GetBackdropColor()
+		column:SetBackdrop(GenericBackdrop)
+		column:SetBackdropColor(r or sr, g or sg, b or sb, a or sa)
+	end
+end
+
+function tipPrototype:SetLineColor(lineNum, r, g, b, a)
+	local line = self.lines[lineNum]
+
+	if line then
+		local sr, sg, sb, sa = self:GetBackdropColor()
+
+		line:SetBackdrop(GenericBackdrop)
+		line:SetBackdropColor(r or sr, g or sg, b or sb, a or sa)
+	end
+end
+
+function tipPrototype:SetCellTextColor(lineNum, colNum, r, g, b, a)
+	local line = self.lines[lineNum]
+	local column = self.columns[colNum]
+
+	if not line or not column then
+		return
+	end
+
+	local cell = self.lines[lineNum].cells[colNum]
+
+	if cell then
+		if not cell.fontString then
+			error("cell's label provider did not assign a fontString field", 2)
+		end
+
+		if not cell.r then
+			cell.r, cell.g, cell.b, cell.a = cell.fontString:GetTextColor()
+		end
+
+		cell.fontString:SetTextColor(r or cell.r, g or cell.g, b or cell.b, a or cell.a)
+	end
+end
+
+function tipPrototype:SetColumnTextColor(colNum, r, g, b, a)
+	if not self.columns[colNum] then
+		return
+	end
+
+	for lineIndex = 1, #self.lines do
+		self:SetCellTextColor(lineIndex, colNum, r, g, b, a)
+	end
+end
+
+function tipPrototype:SetLineTextColor(lineNum, r, g, b, a)
+	local line = self.lines[lineNum]
+
+	if not line then
+		return
+	end
+
+	for cellIndex = 1, #line.cells do
+		self:SetCellTextColor(lineNum, line.cells[cellIndex]._column, r, g, b, a)
+	end
+end
+
+function tipPrototype:SetHighlightTexture(...)
+	return highlightTexture:SetTexture(...)
+end
+
+function tipPrototype:SetHighlightTexCoord(...)
+	highlightTexture:SetTexCoord(...)
+end
+
+do
+	local function checkFont(font, level, silent)
+		local bad = false
+
+		if not font then
+			bad = true
+		elseif type(font) == "string" then
+			local ref = _G[font]
+
+			if not ref or type(ref) ~= "table" or type(ref.IsObjectType) ~= "function" or not ref:IsObjectType("Font") then
+				bad = true
+			end
+		elseif type(font) ~= "table" or type(font.IsObjectType) ~= "function" or not font:IsObjectType("Font") then
+			bad = true
+		end
+
+		if bad then
+			if silent then
+				return false
+			end
+
+			error("font must be a Font instance or a string matching the name of a global Font instance, not: " .. tostring(font), level + 1)
+		end
+		return true
+	end
+
+	function tipPrototype:SetFont(font)
+		local is_string = type(font) == "string"
+
+		checkFont(font, 2)
+		self.regularFont = is_string and _G[font] or font
+	end
+
+	function tipPrototype:SetHeaderFont(font)
+		local is_string = type(font) == "string"
+
+		checkFont(font, 2)
+		self.headerFont = is_string and _G[font] or font
+	end
+
+	-- TODO: fixed argument positions / remove checks for performance?
+	function tipPrototype:SetCell(lineNum, colNum, value, ...)
+		-- Mandatory argument checking
+		if type(lineNum) ~= "number" then
+			error("line number must be a number, not: " .. tostring(lineNum), 2)
+		elseif lineNum < 1 or lineNum > #self.lines then
+			error("line number out of range: " .. tostring(lineNum), 2)
+		elseif type(colNum) ~= "number" then
+			error("column number must be a number, not: " .. tostring(colNum), 2)
+		elseif colNum < 1 or colNum > #self.columns then
+			error("column number out of range: " .. tostring(colNum), 2)
+		end
+
+		-- Variable argument checking
+		local font, justification, colSpan, provider
+		local i, arg = 1, ...
+
+		if arg == nil or checkFont(arg, 2, true) then
+			i, font, arg = 2, ...
+		end
+
+		if arg == nil or checkJustification(arg, 2, true) then
+			i, justification, arg = i + 1, select(i, ...)
+		end
+
+		if arg == nil or type(arg) == "number" then
+			i, colSpan, arg = i + 1, select(i, ...)
+		end
+
+		if arg == nil or type(arg) == "table" and type(arg.AcquireCell) == "function" then
+			i, provider = i + 1, arg
+		end
+
+		return _SetCell(self, lineNum, colNum, value, font, justification, colSpan, provider, select(i, ...))
+	end
+end -- do-block
+
+function tipPrototype:GetFont()
+	return self.regularFont
+end
+
+function tipPrototype:GetHeaderFont()
+	return self.headerFont
+end
+
+function tipPrototype:GetLineCount()
+	return #self.lines
+end
+
+function tipPrototype:GetColumnCount()
+	return #self.columns
+end
+
+------------------------------------------------------------------------------
+-- Frame Scripts
+------------------------------------------------------------------------------
+local scripts = {
+	OnEnter = function(frame, ...)
+		highlightFrame:SetParent(frame)
+		highlightFrame:SetAllPoints(frame)
+		highlightFrame:Show()
+
+		if frame._OnEnter_func then
+			frame:_OnEnter_func(frame._OnEnter_arg, ...)
+		end
+	end,
+	OnLeave = function(frame, ...)
+		highlightFrame:Hide()
+		highlightFrame:ClearAllPoints()
+		highlightFrame:SetParent(nil)
+
+		if frame._OnLeave_func then
+			frame:_OnLeave_func(frame._OnLeave_arg, ...)
+		end
+	end,
+	OnMouseDown = function(frame, ...)
+		frame:_OnMouseDown_func(frame._OnMouseDown_arg, ...)
+	end,
+	OnMouseUp = function(frame, ...)
+		frame:_OnMouseUp_func(frame._OnMouseUp_arg, ...)
+	end,
+	OnReceiveDrag = function(frame, ...)
+		frame:_OnReceiveDrag_func(frame._OnReceiveDrag_arg, ...)
+	end
+}
+
+function SetFrameScript(frame, script, func, arg)
+	if not scripts[script] then
+		return
+	end
+
+	frame["_" .. script .. "_func"] = func
+	frame["_" .. script .. "_arg"] = arg
+
+	if script == "OnMouseDown" or script == "OnMouseUp" or script == "OnReceiveDrag" then
+		if func then
+			frame:SetScript(script, scripts[script])
+		else
+			frame:SetScript(script, nil)
+		end
+	end
+
+	-- if at least one script is set, set the OnEnter/OnLeave scripts for the highlight
+	if frame._OnEnter_func or frame._OnLeave_func or frame._OnMouseDown_func or frame._OnMouseUp_func or frame._OnReceiveDrag_func then
+		frame:EnableMouse(true)
+		frame:SetScript("OnEnter", scripts.OnEnter)
+		frame:SetScript("OnLeave", scripts.OnLeave)
+	else
+		frame:EnableMouse(false)
+		frame:SetScript("OnEnter", nil)
+		frame:SetScript("OnLeave", nil)
+	end
+end
+
+function ClearFrameScripts(frame)
+	if frame._OnEnter_func or frame._OnLeave_func or frame._OnMouseDown_func or frame._OnMouseUp_func or frame._OnReceiveDrag_func then
+		frame:EnableMouse(false)
+
+		frame:SetScript("OnEnter", nil)
+		frame._OnEnter_func = nil
+		frame._OnEnter_arg = nil
+
+		frame:SetScript("OnLeave", nil)
+		frame._OnLeave_func = nil
+		frame._OnLeave_arg = nil
+
+		frame:SetScript("OnReceiveDrag", nil)
+		frame._OnReceiveDrag_func = nil
+		frame._OnReceiveDrag_arg = nil
+
+		frame:SetScript("OnMouseDown", nil)
+		frame._OnMouseDown_func = nil
+		frame._OnMouseDown_arg = nil
+
+		frame:SetScript("OnMouseUp", nil)
+		frame._OnMouseUp_func = nil
+		frame._OnMouseUp_arg = nil
+	end
+end
+
+function tipPrototype:SetLineScript(lineNum, script, func, arg)
+	SetFrameScript(self.lines[lineNum], script, func, arg)
+end
+
+function tipPrototype:SetColumnScript(colNum, script, func, arg)
+	SetFrameScript(self.columns[colNum], script, func, arg)
+end
+
+function tipPrototype:SetCellScript(lineNum, colNum, script, func, arg)
+	local cell = self.lines[lineNum].cells[colNum]
+
+	if cell then
+		SetFrameScript(cell, script, func, arg)
+	end
+end
+
+------------------------------------------------------------------------------
+-- Auto-hiding feature
+------------------------------------------------------------------------------
+
+-- Script of the auto-hiding child frame
+local function AutoHideTimerFrame_OnUpdate(self, elapsed)
+	self.checkElapsed = self.checkElapsed + elapsed
+
+	if self.checkElapsed > 0.1 then
+		if self.parent:IsMouseOver() or (self.alternateFrame and self.alternateFrame:IsMouseOver()) then
+			self.elapsed = 0
+		else
+			self.elapsed = self.elapsed + self.checkElapsed
+
+			if self.elapsed >= self.delay then
+				lib:Release(self.parent)
+			end
+		end
+
+		self.checkElapsed = 0
+	end
+end
+
+-- Usage:
+-- :SetAutoHideDelay(0.25) => hides after 0.25sec outside of the tooltip
+-- :SetAutoHideDelay(0.25, someFrame) => hides after 0.25sec outside of both the tooltip and someFrame
+-- :SetAutoHideDelay() => disable auto-hiding (default)
+function tipPrototype:SetAutoHideDelay(delay, alternateFrame, releaseHandler)
+	local timerFrame = self.autoHideTimerFrame
+	delay = tonumber(delay) or 0
+
+	if releaseHandler then
+		if type(releaseHandler) ~= "function" then
+			error("releaseHandler must be a function", 2)
+		end
+
+		lib.onReleaseHandlers[self] = releaseHandler
+	end
+
+	if delay > 0 then
+		if not timerFrame then
+			timerFrame = AcquireFrame(self)
+			timerFrame:SetScript("OnUpdate", AutoHideTimerFrame_OnUpdate)
+
+			self.autoHideTimerFrame = timerFrame
+		end
+
+		timerFrame.parent = self
+		timerFrame.checkElapsed = 0
+		timerFrame.elapsed = 0
+		timerFrame.delay = delay
+		timerFrame.alternateFrame = alternateFrame
+		timerFrame:Show()
+	elseif timerFrame then
+		self.autoHideTimerFrame = nil
+
+		timerFrame.alternateFrame = nil
+		timerFrame:SetScript("OnUpdate", nil)
+
+		ReleaseFrame(timerFrame)
+	end
+end
+
+------------------------------------------------------------------------------
+-- "Smart" Anchoring
+------------------------------------------------------------------------------
+local function GetTipAnchor(frame)
+	local x, y = frame:GetCenter()
+
+	if not x or not y then
+		return "TOPLEFT", "BOTTOMLEFT"
+	end
+
+	local hhalf = (x > UIParent:GetWidth() * 2 / 3) and "RIGHT" or (x < UIParent:GetWidth() / 3) and "LEFT" or ""
+	local vhalf = (y > UIParent:GetHeight() / 2) and "TOP" or "BOTTOM"
+
+	return vhalf .. hhalf, frame, (vhalf == "TOP" and "BOTTOM" or "TOP") .. hhalf
+end
+
+function tipPrototype:SmartAnchorTo(frame)
+	if not frame then
+		error("Invalid frame provided.", 2)
+	end
+
+	self:ClearAllPoints()
+	self:SetClampedToScreen(true)
+	self:SetPoint(GetTipAnchor(frame))
+end
+
+------------------------------------------------------------------------------
+-- Debug slashcmds
+------------------------------------------------------------------------------
+-- @debug @
+local print = print
+local function PrintStats()
+	local tipCache = tostring(#tooltipHeap)
+	local frameCache = tostring(#frameHeap)
+	local tableCache = tostring(#tableHeap)
+	local header = false
+
+	print("Tooltips used: " .. usedTooltips .. ", Cached: " .. tipCache .. ", Total: " .. tipCache + usedTooltips)
+	print("Frames used: " .. usedFrames .. ", Cached: " .. frameCache .. ", Total: " .. frameCache + usedFrames)
+	print("Tables used: " .. usedTables .. ", Cached: " .. tableCache .. ", Total: " .. tableCache + usedTables)
+
+	for k in pairs(activeTooltips) do
+		if not header then
+			print("Active tooltips:")
+			header = true
+		end
+		print("- " .. k)
+	end
+end
+
+SLASH_LibQTip1 = "/qtip"
+_G.SlashCmdList["LibQTip"] = PrintStats
+--@end-debug@
diff --git a/Titan/libs/LibQTip-1.0/LibQTip-1.0.toc b/Titan/libs/LibQTip-1.0/LibQTip-1.0.toc
new file mode 100755
index 0000000..e9eaac2
--- /dev/null
+++ b/Titan/libs/LibQTip-1.0/LibQTip-1.0.toc
@@ -0,0 +1,14 @@
+## Interface: 80000
+## Title: Lib: QTip-1.0
+## Notes: Library providing multi-column tooltips.
+## Author: Torhal, Adirelle, Elkano, Tristanian
+## Version: 8.0.1.1
+## LoadOnDemand: 1
+## X-Date: 2018-07-21T6:14:41Z
+## X-Credits: Kaelten (input on initial design)
+## X-Category: Library, Tooltip
+## X-License: Ace3 BSD-like license
+## X-Website: http://www.wowace.com/addons/libqtip-1-0/
+
+LibStub\LibStub.lua
+lib.xml
diff --git a/Titan/libs/LibQTip-1.0/LibStub/LibStub.lua b/Titan/libs/LibQTip-1.0/LibStub/LibStub.lua
new file mode 100755
index 0000000..7e7b76d
--- /dev/null
+++ b/Titan/libs/LibQTip-1.0/LibStub/LibStub.lua
@@ -0,0 +1,51 @@
+-- $Id: LibStub.lua 103 2014-10-16 03:02:50Z mikk $
+-- LibStub is a simple versioning stub meant for use in Libraries.  http://www.wowace.com/addons/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]
+
+-- Check to see is this version of the stub is obsolete
+if not LibStub or LibStub.minor < LIBSTUB_MINOR then
+	LibStub = LibStub or {libs = {}, minors = {} }
+	_G[LIBSTUB_MAJOR] = LibStub
+	LibStub.minor = LIBSTUB_MINOR
+
+	-- LibStub:NewLibrary(major, minor)
+	-- major (string) - the major version of the library
+	-- minor (string or number ) - the minor version of the library
+	--
+	-- returns nil if a newer or same version of the lib is already present
+	-- returns empty library object or old library object if upgrade is needed
+	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
+
+	-- LibStub:GetLibrary(major, [silent])
+	-- major (string) - the major version of the library
+	-- silent (boolean) - if true, library is optional, silently return nil if its not found
+	--
+	-- throws an error if the library can not be found (except silent is set)
+	-- returns the library object if found
+	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
+
+	-- LibStub:IterateLibraries()
+	--
+	-- Returns an iterator for the currently registered libraries
+	function LibStub:IterateLibraries()
+		return pairs(self.libs)
+	end
+
+	setmetatable(LibStub, { __call = LibStub.GetLibrary })
+end
diff --git a/Titan/libs/LibQTip-1.0/lib.xml b/Titan/libs/LibQTip-1.0/lib.xml
new file mode 100755
index 0000000..81e11ca
--- /dev/null
+++ b/Titan/libs/LibQTip-1.0/lib.xml
@@ -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="LibQTip-1.0.lua"/>
+</Ui>
diff --git a/Titan/libs/LibSharedMedia-3.0/CHANGES.txt b/Titan/libs/LibSharedMedia-3.0/CHANGES.txt
new file mode 100755
index 0000000..f709a8c
--- /dev/null
+++ b/Titan/libs/LibSharedMedia-3.0/CHANGES.txt
@@ -0,0 +1,8 @@
+------------------------------------------------------------------------
+r102 | funkydude | 2018-07-17 19:25:42 +0000 (Tue, 17 Jul 2018) | 1 line
+Changed paths:
+   M /trunk/LibSharedMedia-3.0.toc
+
+bump version
+------------------------------------------------------------------------
+
diff --git a/Titan/libs/LibSharedMedia-3.0/Changelog-LibSharedMedia-3.0-7.0.3.txt b/Titan/libs/LibSharedMedia-3.0/Changelog-LibSharedMedia-3.0-7.0.3.txt
new file mode 100644
index 0000000..6d2cdc1
--- /dev/null
+++ b/Titan/libs/LibSharedMedia-3.0/Changelog-LibSharedMedia-3.0-7.0.3.txt
@@ -0,0 +1,13 @@
+------------------------------------------------------------------------
+r96 | funkydude | 2016-08-17 19:17:24 +0000 (Wed, 17 Aug 2016) | 1 line
+Changed paths:
+   A /tags/7.0.3 (from /trunk:95)
+
+Tagging as 7.0.3
+------------------------------------------------------------------------
+r95 | funkydude | 2016-07-19 22:45:58 +0000 (Tue, 19 Jul 2016) | 1 line
+Changed paths:
+   M /trunk/LibSharedMedia-3.0.toc
+
+bump toc
+------------------------------------------------------------------------
diff --git a/Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua b/Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua
new file mode 100755
index 0000000..4a31cfe
--- /dev/null
+++ b/Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua
@@ -0,0 +1,292 @@
+--[[
+Name: LibSharedMedia-3.0
+Revision: $Revision: 91 $
+Author: Elkano (elkano@gmx.de)
+Inspired By: SurfaceLib by Haste/Otravi (troeks@gmail.com)
+Website: http://www.wowace.com/projects/libsharedmedia-3-0/
+Description: Shared handling of media data (fonts, sounds, textures, ...) between addons.
+Dependencies: LibStub, CallbackHandler-1.0
+License: LGPL v2.1
+]]
+
+local MAJOR, MINOR = "LibSharedMedia-3.0", 6010002 -- 6.1.0 v2 / increase manually on changes
+local lib = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not lib then return end
+
+local _G = getfenv(0)
+
+local pairs		= _G.pairs
+local type		= _G.type
+
+local band			= _G.bit.band
+
+local table_insert	= _G.table.insert
+local table_sort	= _G.table.sort
+
+local locale = GetLocale()
+local locale_is_western
+local LOCALE_MASK = 0
+lib.LOCALE_BIT_koKR		= 1
+lib.LOCALE_BIT_ruRU		= 2
+lib.LOCALE_BIT_zhCN		= 4
+lib.LOCALE_BIT_zhTW		= 8
+lib.LOCALE_BIT_western	= 128
+
+local CallbackHandler = LibStub:GetLibrary("CallbackHandler-1.0")
+
+lib.callbacks		= lib.callbacks			or CallbackHandler:New(lib)
+
+lib.DefaultMedia	= lib.DefaultMedia		or {}
+lib.MediaList		= lib.MediaList			or {}
+lib.MediaTable		= lib.MediaTable		or {}
+lib.MediaType		= lib.MediaType			or {}
+lib.OverrideMedia	= lib.OverrideMedia		or {}
+
+local defaultMedia = lib.DefaultMedia
+local mediaList = lib.MediaList
+local mediaTable = lib.MediaTable
+local overrideMedia = lib.OverrideMedia
+
+
+-- create mediatype constants
+lib.MediaType.BACKGROUND	= "background"			-- background textures
+lib.MediaType.BORDER		= "border"				-- border textures
+lib.MediaType.FONT			= "font"				-- fonts
+lib.MediaType.STATUSBAR		= "statusbar"			-- statusbar textures
+lib.MediaType.SOUND			= "sound"				-- sound files
+
+-- populate lib with default Blizzard data
+-- BACKGROUND
+if not lib.MediaTable.background then lib.MediaTable.background = {} end
+lib.MediaTable.background["None"]									= [[]]
+lib.MediaTable.background["Blizzard Collections Background"]		= [[Interface\Collections\CollectionsBackgroundTile]]
+lib.MediaTable.background["Blizzard Dialog Background"]				= [[Interface\DialogFrame\UI-DialogBox-Background]]
+lib.MediaTable.background["Blizzard Dialog Background Dark"]		= [[Interface\DialogFrame\UI-DialogBox-Background-Dark]]
+lib.MediaTable.background["Blizzard Dialog Background Gold"]		= [[Interface\DialogFrame\UI-DialogBox-Gold-Background]]
+lib.MediaTable.background["Blizzard Garrison Background"]			= [[Interface\Garrison\GarrisonUIBackground]]
+lib.MediaTable.background["Blizzard Garrison Background 2"]			= [[Interface\Garrison\GarrisonUIBackground2]]
+lib.MediaTable.background["Blizzard Garrison Background 3"]			= [[Interface\Garrison\GarrisonMissionUIInfoBoxBackgroundTile]]
+lib.MediaTable.background["Blizzard Low Health"]					= [[Interface\FullScreenTextures\LowHealth]]
+lib.MediaTable.background["Blizzard Marble"]						= [[Interface\FrameGeneral\UI-Background-Marble]]
+lib.MediaTable.background["Blizzard Out of Control"]				= [[Interface\FullScreenTextures\OutOfControl]]
+lib.MediaTable.background["Blizzard Parchment"]						= [[Interface\AchievementFrame\UI-Achievement-Parchment-Horizontal]]
+lib.MediaTable.background["Blizzard Parchment 2"]					= [[Interface\AchievementFrame\UI-GuildAchievement-Parchment-Horizontal]]
+lib.MediaTable.background["Blizzard Rock"]							= [[Interface\FrameGeneral\UI-Background-Rock]]
+lib.MediaTable.background["Blizzard Tabard Background"]				= [[Interface\TabardFrame\TabardFrameBackground]]
+lib.MediaTable.background["Blizzard Tooltip"]						= [[Interface\Tooltips\UI-Tooltip-Background]]
+lib.MediaTable.background["Solid"]									= [[Interface\Buttons\WHITE8X8]]
+lib.DefaultMedia.background = "None"
+
+-- BORDER
+if not lib.MediaTable.border then lib.MediaTable.border = {} end
+lib.MediaTable.border["None"]								= [[]]
+lib.MediaTable.border["Blizzard Achievement Wood"]			= [[Interface\AchievementFrame\UI-Achievement-WoodBorder]]
+lib.MediaTable.border["Blizzard Chat Bubble"]				= [[Interface\Tooltips\ChatBubble-Backdrop]]
+lib.MediaTable.border["Blizzard Dialog"]					= [[Interface\DialogFrame\UI-DialogBox-Border]]
+lib.MediaTable.border["Blizzard Dialog Gold"]				= [[Interface\DialogFrame\UI-DialogBox-Gold-Border]]
+lib.MediaTable.border["Blizzard Party"]						= [[Interface\CHARACTERFRAME\UI-Party-Border]]
+lib.MediaTable.border["Blizzard Tooltip"]					= [[Interface\Tooltips\UI-Tooltip-Border]]
+lib.DefaultMedia.border = "None"
+
+-- FONT
+if not lib.MediaTable.font then lib.MediaTable.font = {} end
+local SML_MT_font = lib.MediaTable.font
+--[[
+All font files are currently in all clients, the following table depicts which font supports which charset as of 5.0.4
+Fonts were checked using langcover.pl from DejaVu fonts (http://sourceforge.net/projects/dejavu/) and FontForge (http://fontforge.org/)
+latin means check for: de, en, es, fr, it, pt
+
+file				name							latin	koKR	ruRU	zhCN	zhTW
+2002.ttf			2002							X		X		X		-		-
+2002B.ttf			2002 Bold						X		X		X		-		-
+ARHei.ttf			AR CrystalzcuheiGBK Demibold	X		-		X		X		X
+ARIALN.TTF			Arial Narrow					X		-		X		-		-
+ARKai_C.ttf			AR ZhongkaiGBK Medium (Combat)	X		-		X		X		X
+ARKai_T.ttf			AR ZhongkaiGBK Medium			X		-		X		X		X
+bHEI00M.ttf			AR Heiti2 Medium B5				-		-		-		-		X
+bHEI01B.ttf			AR Heiti2 Bold B5				-		-		-		-		X
+bKAI00M.ttf			AR Kaiti Medium B5				-		-		-		-		X
+bLEI00D.ttf			AR Leisu Demi B5				-		-		-		-		X
+FRIZQT__.TTF		Friz Quadrata TT				X		-		-		-		-
+FRIZQT___CYR.TTF	FrizQuadrataCTT					x		-		X		-		-
+K_Damage.TTF		YDIWingsM						-		X		X		-		-
+K_Pagetext.TTF		MoK								X		X		X		-		-
+MORPHEUS.TTF		Morpheus						X		-		-		-		-
+MORPHEUS_CYR.TTF	Morpheus						X		-		X		-		-
+NIM_____.ttf		Nimrod MT						X		-		X		-		-
+SKURRI.TTF			Skurri							X		-		-		-		-
+SKURRI_CYR.TTF		Skurri							X		-		X		-		-
+
+WARNING: Although FRIZQT___CYR is available on western clients, it doesn't support special European characters e.g. é, ï, ö
+Due to this, we cannot use it as a replacement for FRIZQT__.TTF
+]]
+
+if locale == "koKR" then
+	LOCALE_MASK = lib.LOCALE_BIT_koKR
+--
+	SML_MT_font["굵은 글꼴"]		= [[Fonts\2002B.TTF]]
+	SML_MT_font["기본 글꼴"]		= [[Fonts\2002.TTF]]
+	SML_MT_font["데미지 글꼴"]		= [[Fonts\K_Damage.TTF]]
+	SML_MT_font["퀘스트 글꼴"]		= [[Fonts\K_Pagetext.TTF]]
+--
+	lib.DefaultMedia["font"] = "기본 글꼴" -- someone from koKR please adjust if needed
+--
+elseif locale == "zhCN" then
+	LOCALE_MASK = lib.LOCALE_BIT_zhCN
+--
+	SML_MT_font["伤害数字"]		= [[Fonts\ARKai_C.ttf]]
+	SML_MT_font["默认"]			= [[Fonts\ARKai_T.ttf]]
+	SML_MT_font["聊天"]			= [[Fonts\ARHei.ttf]]
+--
+	lib.DefaultMedia["font"] = "默认" -- someone from zhCN please adjust if needed
+--
+elseif locale == "zhTW" then
+	LOCALE_MASK = lib.LOCALE_BIT_zhTW
+--
+	SML_MT_font["提示訊息"]		= [[Fonts\bHEI00M.ttf]]
+	SML_MT_font["聊天"]			= [[Fonts\bHEI01B.ttf]]
+	SML_MT_font["傷害數字"]		= [[Fonts\bKAI00M.ttf]]
+	SML_MT_font["預設"]			= [[Fonts\bLEI00D.ttf]]
+--
+	lib.DefaultMedia["font"] = "預設" -- someone from zhTW please adjust if needed
+
+elseif locale == "ruRU" then
+	LOCALE_MASK = lib.LOCALE_BIT_ruRU
+--
+	SML_MT_font["2002"]								= [[Fonts\2002.TTF]]
+	SML_MT_font["2002 Bold"]						= [[Fonts\2002B.TTF]]
+	SML_MT_font["AR CrystalzcuheiGBK Demibold"]		= [[Fonts\ARHei.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium (Combat)"]	= [[Fonts\ARKai_C.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium"]			= [[Fonts\ARKai_T.TTF]]
+	SML_MT_font["Arial Narrow"]						= [[Fonts\ARIALN.TTF]]
+	SML_MT_font["Friz Quadrata TT"]					= [[Fonts\FRIZQT___CYR.TTF]]
+	SML_MT_font["MoK"]								= [[Fonts\K_Pagetext.TTF]]
+	SML_MT_font["Morpheus"]							= [[Fonts\MORPHEUS_CYR.TTF]]
+	SML_MT_font["Nimrod MT"]						= [[Fonts\NIM_____.ttf]]
+	SML_MT_font["Skurri"]							= [[Fonts\SKURRI_CYR.TTF]]
+--
+	lib.DefaultMedia.font = "Friz Quadrata TT"
+--
+else
+	LOCALE_MASK = lib.LOCALE_BIT_western
+	locale_is_western = true
+--
+	SML_MT_font["2002"]								= [[Fonts\2002.TTF]]
+	SML_MT_font["2002 Bold"]						= [[Fonts\2002B.TTF]]
+	SML_MT_font["AR CrystalzcuheiGBK Demibold"]		= [[Fonts\ARHei.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium (Combat)"]	= [[Fonts\ARKai_C.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium"]			= [[Fonts\ARKai_T.TTF]]
+	SML_MT_font["Arial Narrow"]						= [[Fonts\ARIALN.TTF]]
+	SML_MT_font["Friz Quadrata TT"]					= [[Fonts\FRIZQT__.TTF]]
+	SML_MT_font["MoK"]								= [[Fonts\K_Pagetext.TTF]]
+	SML_MT_font["Morpheus"]							= [[Fonts\MORPHEUS_CYR.TTF]]
+	SML_MT_font["Nimrod MT"]						= [[Fonts\NIM_____.ttf]]
+	SML_MT_font["Skurri"]							= [[Fonts\SKURRI_CYR.TTF]]
+--
+	lib.DefaultMedia.font = "Friz Quadrata TT"
+--
+end
+
+-- STATUSBAR
+if not lib.MediaTable.statusbar then lib.MediaTable.statusbar = {} end
+lib.MediaTable.statusbar["Blizzard"]						= [[Interface\TargetingFrame\UI-StatusBar]]
+lib.MediaTable.statusbar["Blizzard Character Skills Bar"]	= [[Interface\PaperDollInfoFrame\UI-Character-Skills-Bar]]
+lib.MediaTable.statusbar["Blizzard Raid Bar"]				= [[Interface\RaidFrame\Raid-Bar-Hp-Fill]]
+lib.DefaultMedia.statusbar = "Blizzard"
+
+-- SOUND
+if not lib.MediaTable.sound then lib.MediaTable.sound = {} end
+lib.MediaTable.sound["None"]								= [[Interface\Quiet.ogg]]	-- Relies on the fact that PlaySound[File] doesn't error on non-existing input.
+lib.DefaultMedia.sound = "None"
+
+local function rebuildMediaList(mediatype)
+	local mtable = mediaTable[mediatype]
+	if not mtable then return end
+	if not mediaList[mediatype] then mediaList[mediatype] = {} end
+	local mlist = mediaList[mediatype]
+	-- list can only get larger, so simply overwrite it
+	local i = 0
+	for k in pairs(mtable) do
+		i = i + 1
+		mlist[i] = k
+	end
+	table_sort(mlist)
+end
+
+function lib:Register(mediatype, key, data, langmask)
+	if type(mediatype) ~= "string" then
+		error(MAJOR..":Register(mediatype, key, data, langmask) - mediatype must be string, got "..type(mediatype))
+	end
+	if type(key) ~= "string" then
+		error(MAJOR..":Register(mediatype, key, data, langmask) - key must be string, got "..type(key))
+	end
+	mediatype = mediatype:lower()
+	if mediatype == lib.MediaType.FONT and ((langmask and band(langmask, LOCALE_MASK) == 0) or not (langmask or locale_is_western)) then return false end
+	if mediatype == lib.MediaType.SOUND and type(data) == "string" then
+		local path = data:lower()
+		-- Only ogg and mp3 are valid sounds.
+		if not path:find(".ogg", nil, true) and not path:find(".mp3", nil, true) then
+			return false
+		end
+	end
+	if not mediaTable[mediatype] then mediaTable[mediatype] = {} end
+	local mtable = mediaTable[mediatype]
+	if mtable[key] then return false end
+
+	mtable[key] = data
+	rebuildMediaList(mediatype)
+	self.callbacks:Fire("LibSharedMedia_Registered", mediatype, key)
+	return true
+end
+
+function lib:Fetch(mediatype, key, noDefault)
+	local mtt = mediaTable[mediatype]
+	local overridekey = overrideMedia[mediatype]
+	local result = mtt and ((overridekey and mtt[overridekey] or mtt[key]) or (not noDefault and defaultMedia[mediatype] and mtt[defaultMedia[mediatype]])) or nil
+	return result ~= "" and result or nil
+end
+
+function lib:IsValid(mediatype, key)
+	return mediaTable[mediatype] and (not key or mediaTable[mediatype][key]) and true or false
+end
+
+function lib:HashTable(mediatype)
+	return mediaTable[mediatype]
+end
+
+function lib:List(mediatype)
+	if not mediaTable[mediatype] then
+		return nil
+	end
+	if not mediaList[mediatype] then
+		rebuildMediaList(mediatype)
+	end
+	return mediaList[mediatype]
+end
+
+function lib:GetGlobal(mediatype)
+	return overrideMedia[mediatype]
+end
+
+function lib:SetGlobal(mediatype, key)
+	if not mediaTable[mediatype] then
+		return false
+	end
+	overrideMedia[mediatype] = (key and mediaTable[mediatype][key]) and key or nil
+	self.callbacks:Fire("LibSharedMedia_SetGlobal", mediatype, overrideMedia[mediatype])
+	return true
+end
+
+function lib:GetDefault(mediatype)
+	return defaultMedia[mediatype]
+end
+
+function lib:SetDefault(mediatype, key)
+	if mediaTable[mediatype] and mediaTable[mediatype][key] and not defaultMedia[mediatype] then
+		defaultMedia[mediatype] = key
+		return true
+	else
+		return false
+	end
+end
diff --git a/Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.toc b/Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.toc
new file mode 100755
index 0000000..2a1f890
--- /dev/null
+++ b/Titan/libs/LibSharedMedia-3.0/LibSharedMedia-3.0.toc
@@ -0,0 +1,18 @@
+## Interface: 80000
+## LoadOnDemand: 1
+
+## Title: Lib: SharedMedia-3.0
+## Notes: Shared handling of media data (fonts, sounds, textures, ...) between addons.
+## Author: Elkano
+## Version: 3.0-102
+## X-Website: http://www.wowace.com/projects/libsharedmedia-3-0/
+## X-Category: Library
+
+## X-Revision: 102
+## X-Date: 2018-07-17T19:25:42Z
+
+LibStub\LibStub.lua
+CallbackHandler-1.0\CallbackHandler-1.0.lua
+
+LibSharedMedia-3.0\lib.xml
+
diff --git a/Titan/libs/LibSharedMedia-3.0/lib.xml b/Titan/libs/LibSharedMedia-3.0/lib.xml
new file mode 100755
index 0000000..7313228
--- /dev/null
+++ b/Titan/libs/LibSharedMedia-3.0/lib.xml
@@ -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="LibSharedMedia-3.0.lua" />
+</Ui>
diff --git a/Titan/libs/LibStub/LibStub.lua b/Titan/libs/LibStub/LibStub.lua
new file mode 100755
index 0000000..f5fc919
--- /dev/null
+++ b/Titan/libs/LibStub/LibStub.lua
@@ -0,0 +1,51 @@
+-- $Id: LibStub.lua 76 2007-09-03 01:50:17Z mikk $
+-- 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]
+
+-- Check to see is this version of the stub is obsolete
+if not LibStub or LibStub.minor < LIBSTUB_MINOR then
+	LibStub = LibStub or {libs = {}, minors = {} }
+	_G[LIBSTUB_MAJOR] = LibStub
+	LibStub.minor = LIBSTUB_MINOR
+
+	-- LibStub:NewLibrary(major, minor)
+	-- major (string) - the major version of the library
+	-- minor (string or number ) - the minor version of the library
+	--
+	-- returns nil if a newer or same version of the lib is already present
+	-- returns empty library object or old library object if upgrade is needed
+	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
+
+	-- LibStub:GetLibrary(major, [silent])
+	-- major (string) - the major version of the library
+	-- silent (boolean) - if true, library is optional, silently return nil if its not found
+	--
+	-- throws an error if the library can not be found (except silent is set)
+	-- returns the library object if found
+	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
+
+	-- LibStub:IterateLibraries()
+	--
+	-- Returns an iterator for the currently registered libraries
+	function LibStub:IterateLibraries()
+		return pairs(self.libs)
+	end
+
+	setmetatable(LibStub, { __call = LibStub.GetLibrary })
+end
diff --git a/Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.lua b/Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.lua
new file mode 100644
index 0000000..5ebb35d
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.lua
@@ -0,0 +1,813 @@
+--- **AceGUI-3.0** provides access to numerous widgets which can be used to create GUIs.
+-- AceGUI is used by AceConfigDialog to create the option GUIs, but you can use it by itself
+-- to create any custom GUI. There are more extensive examples in the test suite in the Ace3
+-- stand-alone distribution.
+--
+-- **Note**: When using AceGUI-3.0 directly, please do not modify the frames of the widgets directly,
+-- as any "unknown" change to the widgets will cause addons that get your widget out of the widget pool
+-- to misbehave. If you think some part of a widget should be modifiable, please open a ticket, and we"ll
+-- implement a proper API to modify it.
+-- @usage
+-- local AceGUI = LibStub("AceGUI-3.0")
+-- -- Create a container frame
+-- local f = AceGUI:Create("Frame")
+-- f:SetCallback("OnClose",function(widget) AceGUI:Release(widget) end)
+-- f:SetTitle("AceGUI-3.0 Example")
+-- f:SetStatusText("Status Bar")
+-- f:SetLayout("Flow")
+-- -- Create a button
+-- local btn = AceGUI:Create("Button")
+-- btn:SetWidth(170)
+-- btn:SetText("Button !")
+-- btn:SetCallback("OnClick", function() print("Click!") end)
+-- -- Add the button to the container
+-- f:AddChild(btn)
+-- @class file
+-- @name AceGUI-3.0
+-- @release $Id: AceGUI-3.0.lua 1102 2013-10-25 14:15:23Z nevcairiel $
+local ACEGUI_MAJOR, ACEGUI_MINOR = "AceGUI-3.0", 34
+local AceGUI, oldminor = LibStub:NewLibrary(ACEGUI_MAJOR, ACEGUI_MINOR)
+
+if not AceGUI then return end -- No upgrade needed
+
+-- Lua APIs
+local tconcat, tremove, tinsert = table.concat, table.remove, table.insert
+local select, pairs, next, type = select, pairs, next, type
+local error, assert, loadstring = error, assert, loadstring
+local setmetatable, rawget, rawset = setmetatable, rawget, rawset
+local math_max = math.max
+
+-- WoW APIs
+local UIParent = 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: geterrorhandler, LibStub
+
+--local con = LibStub("AceConsole-3.0",true)
+
+AceGUI.WidgetRegistry = AceGUI.WidgetRegistry or {}
+AceGUI.LayoutRegistry = AceGUI.LayoutRegistry or {}
+AceGUI.WidgetBase = AceGUI.WidgetBase or {}
+AceGUI.WidgetContainerBase = AceGUI.WidgetContainerBase or {}
+AceGUI.WidgetVersions = AceGUI.WidgetVersions or {}
+
+-- local upvalues
+local WidgetRegistry = AceGUI.WidgetRegistry
+local LayoutRegistry = AceGUI.LayoutRegistry
+local WidgetVersions = AceGUI.WidgetVersions
+
+--[[
+	 xpcall safecall implementation
+]]
+local xpcall = xpcall
+
+local function errorhandler(err)
+	return geterrorhandler()(err)
+end
+
+local function CreateDispatcher(argCount)
+	local code = [[
+		local xpcall, eh = ...
+		local method, ARGS
+		local function call() return method(ARGS) end
+
+		local function dispatch(func, ...)
+			method = func
+			if not method then return end
+			ARGS = ...
+			return xpcall(call, eh)
+		end
+
+		return dispatch
+	]]
+
+	local ARGS = {}
+	for i = 1, argCount do ARGS[i] = "arg"..i end
+	code = code:gsub("ARGS", tconcat(ARGS, ", "))
+	return assert(loadstring(code, "safecall Dispatcher["..argCount.."]"))(xpcall, errorhandler)
+end
+
+local Dispatchers = setmetatable({}, {__index=function(self, argCount)
+	local dispatcher = CreateDispatcher(argCount)
+	rawset(self, argCount, dispatcher)
+	return dispatcher
+end})
+Dispatchers[0] = function(func)
+	return xpcall(func, errorhandler)
+end
+
+local function safecall(func, ...)
+	return Dispatchers[select("#", ...)](func, ...)
+end
+
+-- Recycling functions
+local newWidget, delWidget
+do
+	-- Version Upgrade in Minor 29
+	-- Internal Storage of the objects changed, from an array table
+	-- to a hash table, and additionally we introduced versioning on
+	-- the widgets which would discard all widgets from a pre-29 version
+	-- anyway, so we just clear the storage now, and don't try to
+	-- convert the storage tables to the new format.
+	-- This should generally not cause *many* widgets to end up in trash,
+	-- since once dialogs are opened, all addons should be loaded already
+	-- and AceGUI should be on the latest version available on the users
+	-- setup.
+	-- -- nevcairiel - Nov 2nd, 2009
+	if oldminor and oldminor < 29 and AceGUI.objPools then
+		AceGUI.objPools = nil
+	end
+
+	AceGUI.objPools = AceGUI.objPools or {}
+	local objPools = AceGUI.objPools
+	--Returns a new instance, if none are available either returns a new table or calls the given contructor
+	function newWidget(type)
+		if not WidgetRegistry[type] then
+			error("Attempt to instantiate unknown widget type", 2)
+		end
+
+		if not objPools[type] then
+			objPools[type] = {}
+		end
+
+		local newObj = next(objPools[type])
+		if not newObj then
+			newObj = WidgetRegistry[type]()
+			newObj.AceGUIWidgetVersion = WidgetVersions[type]
+		else
+			objPools[type][newObj] = nil
+			-- if the widget is older then the latest, don't even try to reuse it
+			-- just forget about it, and grab a new one.
+			if not newObj.AceGUIWidgetVersion or newObj.AceGUIWidgetVersion < WidgetVersions[type] then
+				return newWidget(type)
+			end
+		end
+		return newObj
+	end
+	-- Releases an instance to the Pool
+	function delWidget(obj,type)
+		if not objPools[type] then
+			objPools[type] = {}
+		end
+		if objPools[type][obj] then
+			error("Attempt to Release Widget that is already released", 2)
+		end
+		objPools[type][obj] = true
+	end
+end
+
+
+-------------------
+-- API Functions --
+-------------------
+
+-- Gets a widget Object
+
+--- Create a new Widget of the given type.
+-- This function will instantiate a new widget (or use one from the widget pool), and call the
+-- OnAcquire function on it, before returning.
+-- @param type The type of the widget.
+-- @return The newly created widget.
+function AceGUI:Create(type)
+	if WidgetRegistry[type] then
+		local widget = newWidget(type)
+
+		if rawget(widget, "Acquire") then
+			widget.OnAcquire = widget.Acquire
+			widget.Acquire = nil
+		elseif rawget(widget, "Aquire") then
+			widget.OnAcquire = widget.Aquire
+			widget.Aquire = nil
+		end
+
+		if rawget(widget, "Release") then
+			widget.OnRelease = rawget(widget, "Release")
+			widget.Release = nil
+		end
+
+		if widget.OnAcquire then
+			widget:OnAcquire()
+		else
+			error(("Widget type %s doesn't supply an OnAcquire Function"):format(type))
+		end
+		-- Set the default Layout ("List")
+		safecall(widget.SetLayout, widget, "List")
+		safecall(widget.ResumeLayout, widget)
+		return widget
+	end
+end
+
+--- Releases a widget Object.
+-- This function calls OnRelease on the widget and places it back in the widget pool.
+-- Any data on the widget is being erased, and the widget will be hidden.\\
+-- If this widget is a Container-Widget, all of its Child-Widgets will be releases as well.
+-- @param widget The widget to release
+function AceGUI:Release(widget)
+	safecall(widget.PauseLayout, widget)
+	widget:Fire("OnRelease")
+	safecall(widget.ReleaseChildren, widget)
+
+	if widget.OnRelease then
+		widget:OnRelease()
+--	else
+--		error(("Widget type %s doesn't supply an OnRelease Function"):format(widget.type))
+	end
+	for k in pairs(widget.userdata) do
+		widget.userdata[k] = nil
+	end
+	for k in pairs(widget.events) do
+		widget.events[k] = nil
+	end
+	widget.width = nil
+	widget.relWidth = nil
+	widget.height = nil
+	widget.relHeight = nil
+	widget.noAutoHeight = nil
+	widget.frame:ClearAllPoints()
+	widget.frame:Hide()
+	widget.frame:SetParent(UIParent)
+	widget.frame.width = nil
+	widget.frame.height = nil
+	if widget.content then
+		widget.content.width = nil
+		widget.content.height = nil
+	end
+	delWidget(widget, widget.type)
+end
+
+-----------
+-- Focus --
+-----------
+
+
+--- Called when a widget has taken focus.
+-- e.g. Dropdowns opening, Editboxes gaining kb focus
+-- @param widget The widget that should be focused
+function AceGUI:SetFocus(widget)
+	if self.FocusedWidget and self.FocusedWidget ~= widget then
+		safecall(self.FocusedWidget.ClearFocus, self.FocusedWidget)
+	end
+	self.FocusedWidget = widget
+end
+
+
+--- Called when something has happened that could cause widgets with focus to drop it
+-- e.g. titlebar of a frame being clicked
+function AceGUI:ClearFocus()
+	if self.FocusedWidget then
+		safecall(self.FocusedWidget.ClearFocus, self.FocusedWidget)
+		self.FocusedWidget = nil
+	end
+end
+
+-------------
+-- Widgets --
+-------------
+--[[
+	Widgets must provide the following functions
+		OnAcquire() - Called when the object is acquired, should set everything to a default hidden state
+
+	And the following members
+		frame - the frame or derivitive object that will be treated as the widget for size and anchoring purposes
+		type - the type of the object, same as the name given to :RegisterWidget()
+
+	Widgets contain a table called userdata, this is a safe place to store data associated with the wigdet
+	It will be cleared automatically when a widget is released
+	Placing values directly into a widget object should be avoided
+
+	If the Widget can act as a container for other Widgets the following
+		content - frame or derivitive that children will be anchored to
+
+	The Widget can supply the following Optional Members
+		:OnRelease() - Called when the object is Released, should remove any additional anchors and clear any data
+		:OnWidthSet(width) - Called when the width of the widget is changed
+		:OnHeightSet(height) - Called when the height of the widget is changed
+			Widgets should not use the OnSizeChanged events of thier frame or content members, use these methods instead
+			AceGUI already sets a handler to the event
+		:LayoutFinished(width, height) - called after a layout has finished, the width and height will be the width and height of the
+			area used for controls. These can be nil if the layout used the existing size to layout the controls.
+
+]]
+
+--------------------------
+-- Widget Base Template --
+--------------------------
+do
+	local WidgetBase = AceGUI.WidgetBase
+
+	WidgetBase.SetParent = function(self, parent)
+		local frame = self.frame
+		frame:SetParent(nil)
+		frame:SetParent(parent.content)
+		self.parent = parent
+	end
+
+	WidgetBase.SetCallback = function(self, name, func)
+		if type(func) == "function" then
+			self.events[name] = func
+		end
+	end
+
+	WidgetBase.Fire = function(self, name, ...)
+		if self.events[name] then
+			local success, ret = safecall(self.events[name], self, name, ...)
+			if success then
+				return ret
+			end
+		end
+	end
+
+	WidgetBase.SetWidth = function(self, width)
+		self.frame:SetWidth(width)
+		self.frame.width = width
+		if self.OnWidthSet then
+			self:OnWidthSet(width)
+		end
+	end
+
+	WidgetBase.SetRelativeWidth = function(self, width)
+		if width <= 0 or width > 1 then
+			error(":SetRelativeWidth(width): Invalid relative width.", 2)
+		end
+		self.relWidth = width
+		self.width = "relative"
+	end
+
+	WidgetBase.SetHeight = function(self, height)
+		self.frame:SetHeight(height)
+		self.frame.height = height
+		if self.OnHeightSet then
+			self:OnHeightSet(height)
+		end
+	end
+
+	--[[ WidgetBase.SetRelativeHeight = function(self, height)
+		if height <= 0 or height > 1 then
+			error(":SetRelativeHeight(height): Invalid relative height.", 2)
+		end
+		self.relHeight = height
+		self.height = "relative"
+	end ]]
+
+	WidgetBase.IsVisible = function(self)
+		return self.frame:IsVisible()
+	end
+
+	WidgetBase.IsShown= function(self)
+		return self.frame:IsShown()
+	end
+
+	WidgetBase.Release = function(self)
+		AceGUI:Release(self)
+	end
+
+	WidgetBase.SetPoint = function(self, ...)
+		return self.frame:SetPoint(...)
+	end
+
+	WidgetBase.ClearAllPoints = function(self)
+		return self.frame:ClearAllPoints()
+	end
+
+	WidgetBase.GetNumPoints = function(self)
+		return self.frame:GetNumPoints()
+	end
+
+	WidgetBase.GetPoint = function(self, ...)
+		return self.frame:GetPoint(...)
+	end
+
+	WidgetBase.GetUserDataTable = function(self)
+		return self.userdata
+	end
+
+	WidgetBase.SetUserData = function(self, key, value)
+		self.userdata[key] = value
+	end
+
+	WidgetBase.GetUserData = function(self, key)
+		return self.userdata[key]
+	end
+
+	WidgetBase.IsFullHeight = function(self)
+		return self.height == "fill"
+	end
+
+	WidgetBase.SetFullHeight = function(self, isFull)
+		if isFull then
+			self.height = "fill"
+		else
+			self.height = nil
+		end
+	end
+
+	WidgetBase.IsFullWidth = function(self)
+		return self.width == "fill"
+	end
+
+	WidgetBase.SetFullWidth = function(self, isFull)
+		if isFull then
+			self.width = "fill"
+		else
+			self.width = nil
+		end
+	end
+
+--	local function LayoutOnUpdate(this)
+--		this:SetScript("OnUpdate",nil)
+--		this.obj:PerformLayout()
+--	end
+
+	local WidgetContainerBase = AceGUI.WidgetContainerBase
+
+	WidgetContainerBase.PauseLayout = function(self)
+		self.LayoutPaused = true
+	end
+
+	WidgetContainerBase.ResumeLayout = function(self)
+		self.LayoutPaused = nil
+	end
+
+	WidgetContainerBase.PerformLayout = function(self)
+		if self.LayoutPaused then
+			return
+		end
+		safecall(self.LayoutFunc, self.content, self.children)
+	end
+
+	--call this function to layout, makes sure layed out objects get a frame to get sizes etc
+	WidgetContainerBase.DoLayout = function(self)
+		self:PerformLayout()
+--		if not self.parent then
+--			self.frame:SetScript("OnUpdate", LayoutOnUpdate)
+--		end
+	end
+
+	WidgetContainerBase.AddChild = function(self, child, beforeWidget)
+		if beforeWidget then
+			local siblingIndex = 1
+			for _, widget in pairs(self.children) do
+				if widget == beforeWidget then
+					break
+				end
+				siblingIndex = siblingIndex + 1
+			end
+			tinsert(self.children, siblingIndex, child)
+		else
+			tinsert(self.children, child)
+		end
+		child:SetParent(self)
+		child.frame:Show()
+		self:DoLayout()
+	end
+
+	WidgetContainerBase.AddChildren = function(self, ...)
+		for i = 1, select("#", ...) do
+			local child = select(i, ...)
+			tinsert(self.children, child)
+			child:SetParent(self)
+			child.frame:Show()
+		end
+		self:DoLayout()
+	end
+
+	WidgetContainerBase.ReleaseChildren = function(self)
+		local children = self.children
+		for i = 1,#children do
+			AceGUI:Release(children[i])
+			children[i] = nil
+		end
+	end
+
+	WidgetContainerBase.SetLayout = function(self, Layout)
+		self.LayoutFunc = AceGUI:GetLayout(Layout)
+	end
+
+	WidgetContainerBase.SetAutoAdjustHeight = function(self, adjust)
+		if adjust then
+			self.noAutoHeight = nil
+		else
+			self.noAutoHeight = true
+		end
+	end
+
+	local function FrameResize(this)
+		local self = this.obj
+		if this:GetWidth() and this:GetHeight() then
+			if self.OnWidthSet then
+				self:OnWidthSet(this:GetWidth())
+			end
+			if self.OnHeightSet then
+				self:OnHeightSet(this:GetHeight())
+			end
+		end
+	end
+
+	local function ContentResize(this)
+		if this:GetWidth() and this:GetHeight() then
+			this.width = this:GetWidth()
+			this.height = this:GetHeight()
+			this.obj:DoLayout()
+		end
+	end
+
+	setmetatable(WidgetContainerBase, {__index=WidgetBase})
+
+	--One of these function should be called on each Widget Instance as part of its creation process
+
+	--- Register a widget-class as a container for newly created widgets.
+	-- @param widget The widget class
+	function AceGUI:RegisterAsContainer(widget)
+		widget.children = {}
+		widget.userdata = {}
+		widget.events = {}
+		widget.base = WidgetContainerBase
+		widget.content.obj = widget
+		widget.frame.obj = widget
+		widget.content:SetScript("OnSizeChanged", ContentResize)
+		widget.frame:SetScript("OnSizeChanged", FrameResize)
+		setmetatable(widget, {__index = WidgetContainerBase})
+		widget:SetLayout("List")
+		return widget
+	end
+
+	--- Register a widget-class as a widget.
+	-- @param widget The widget class
+	function AceGUI:RegisterAsWidget(widget)
+		widget.userdata = {}
+		widget.events = {}
+		widget.base = WidgetBase
+		widget.frame.obj = widget
+		widget.frame:SetScript("OnSizeChanged", FrameResize)
+		setmetatable(widget, {__index = WidgetBase})
+		return widget
+	end
+end
+
+
+
+
+------------------
+-- Widget API   --
+------------------
+
+--- Registers a widget Constructor, this function returns a new instance of the Widget
+-- @param Name The name of the widget
+-- @param Constructor The widget constructor function
+-- @param Version The version of the widget
+function AceGUI:RegisterWidgetType(Name, Constructor, Version)
+	assert(type(Constructor) == "function")
+	assert(type(Version) == "number")
+
+	local oldVersion = WidgetVersions[Name]
+	if oldVersion and oldVersion >= Version then return end
+
+	WidgetVersions[Name] = Version
+	WidgetRegistry[Name] = Constructor
+end
+
+--- Registers a Layout Function
+-- @param Name The name of the layout
+-- @param LayoutFunc Reference to the layout function
+function AceGUI:RegisterLayout(Name, LayoutFunc)
+	assert(type(LayoutFunc) == "function")
+	if type(Name) == "string" then
+		Name = Name:upper()
+	end
+	LayoutRegistry[Name] = LayoutFunc
+end
+
+--- Get a Layout Function from the registry
+-- @param Name The name of the layout
+function AceGUI:GetLayout(Name)
+	if type(Name) == "string" then
+		Name = Name:upper()
+	end
+	return LayoutRegistry[Name]
+end
+
+AceGUI.counts = AceGUI.counts or {}
+
+--- A type-based counter to count the number of widgets created.
+-- This is used by widgets that require a named frame, e.g. when a Blizzard
+-- Template requires it.
+-- @param type The widget type
+function AceGUI:GetNextWidgetNum(type)
+	if not self.counts[type] then
+		self.counts[type] = 0
+	end
+	self.counts[type] = self.counts[type] + 1
+	return self.counts[type]
+end
+
+--- Return the number of created widgets for this type.
+-- In contrast to GetNextWidgetNum, the number is not incremented.
+-- @param type The widget type
+function AceGUI:GetWidgetCount(type)
+	return self.counts[type] or 0
+end
+
+--- Return the version of the currently registered widget type.
+-- @param type The widget type
+function AceGUI:GetWidgetVersion(type)
+	return WidgetVersions[type]
+end
+
+-------------
+-- Layouts --
+-------------
+
+--[[
+	A Layout is a func that takes 2 parameters
+		content - the frame that widgets will be placed inside
+		children - a table containing the widgets to layout
+]]
+
+-- Very simple Layout, Children are stacked on top of each other down the left side
+AceGUI:RegisterLayout("List",
+	function(content, children)
+		local height = 0
+		local width = content.width or content:GetWidth() or 0
+		for i = 1, #children do
+			local child = children[i]
+
+			local frame = child.frame
+			frame:ClearAllPoints()
+			frame:Show()
+			if i == 1 then
+				frame:SetPoint("TOPLEFT", content)
+			else
+				frame:SetPoint("TOPLEFT", children[i-1].frame, "BOTTOMLEFT")
+			end
+
+			if child.width == "fill" then
+				child:SetWidth(width)
+				frame:SetPoint("RIGHT", content)
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+			elseif child.width == "relative" then
+				child:SetWidth(width * child.relWidth)
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+			end
+
+			height = height + (frame.height or frame:GetHeight() or 0)
+		end
+		safecall(content.obj.LayoutFinished, content.obj, nil, height)
+	end)
+
+-- A single control fills the whole content area
+AceGUI:RegisterLayout("Fill",
+	function(content, children)
+		if children[1] then
+			children[1]:SetWidth(content:GetWidth() or 0)
+			children[1]:SetHeight(content:GetHeight() or 0)
+			children[1].frame:SetAllPoints(content)
+			children[1].frame:Show()
+			safecall(content.obj.LayoutFinished, content.obj, nil, children[1].frame:GetHeight())
+		end
+	end)
+
+local layoutrecursionblock = nil
+local function safelayoutcall(object, func, ...)
+	layoutrecursionblock = true
+	object[func](object, ...)
+	layoutrecursionblock = nil
+end
+
+AceGUI:RegisterLayout("Flow",
+	function(content, children)
+		if layoutrecursionblock then return end
+		--used height so far
+		local height = 0
+		--width used in the current row
+		local usedwidth = 0
+		--height of the current row
+		local rowheight = 0
+		local rowoffset = 0
+		local lastrowoffset
+
+		local width = content.width or content:GetWidth() or 0
+
+		--control at the start of the row
+		local rowstart
+		local rowstartoffset
+		local lastrowstart
+		local isfullheight
+
+		local frameoffset
+		local lastframeoffset
+		local oversize
+		for i = 1, #children do
+			local child = children[i]
+			oversize = nil
+			local frame = child.frame
+			local frameheight = frame.height or frame:GetHeight() or 0
+			local framewidth = frame.width or frame:GetWidth() or 0
+			lastframeoffset = frameoffset
+			-- HACK: Why did we set a frameoffset of (frameheight / 2) ?
+			-- That was moving all widgets half the widgets size down, is that intended?
+			-- Actually, it seems to be neccessary for many cases, we'll leave it in for now.
+			-- If widgets seem to anchor weirdly with this, provide a valid alignoffset for them.
+			-- TODO: Investigate moar!
+			frameoffset = child.alignoffset or (frameheight / 2)
+
+			if child.width == "relative" then
+				framewidth = width * child.relWidth
+			end
+
+			frame:Show()
+			frame:ClearAllPoints()
+			if i == 1 then
+				-- anchor the first control to the top left
+				frame:SetPoint("TOPLEFT", content)
+				rowheight = frameheight
+				rowoffset = frameoffset
+				rowstart = frame
+				rowstartoffset = frameoffset
+				usedwidth = framewidth
+				if usedwidth > width then
+					oversize = true
+				end
+			else
+				-- if there isn't available width for the control start a new row
+				-- if a control is "fill" it will be on a row of its own full width
+				if usedwidth == 0 or ((framewidth) + usedwidth > width) or child.width == "fill" then
+					if isfullheight then
+						-- a previous row has already filled the entire height, there's nothing we can usefully do anymore
+						-- (maybe error/warn about this?)
+						break
+					end
+					--anchor the previous row, we will now know its height and offset
+					rowstart:SetPoint("TOPLEFT", content, "TOPLEFT", 0, -(height + (rowoffset - rowstartoffset) + 3))
+					height = height + rowheight + 3
+					--save this as the rowstart so we can anchor it after the row is complete and we have the max height and offset of controls in it
+					rowstart = frame
+					rowstartoffset = frameoffset
+					rowheight = frameheight
+					rowoffset = frameoffset
+					usedwidth = framewidth
+					if usedwidth > width then
+						oversize = true
+					end
+				-- put the control on the current row, adding it to the width and checking if the height needs to be increased
+				else
+					--handles cases where the new height is higher than either control because of the offsets
+					--math.max(rowheight-rowoffset+frameoffset, frameheight-frameoffset+rowoffset)
+
+					--offset is always the larger of the two offsets
+					rowoffset = math_max(rowoffset, frameoffset)
+					rowheight = math_max(rowheight, rowoffset + (frameheight / 2))
+
+					frame:SetPoint("TOPLEFT", children[i-1].frame, "TOPRIGHT", 0, frameoffset - lastframeoffset)
+					usedwidth = framewidth + usedwidth
+				end
+			end
+
+			if child.width == "fill" then
+				safelayoutcall(child, "SetWidth", width)
+				frame:SetPoint("RIGHT", content)
+
+				usedwidth = 0
+				rowstart = frame
+				rowstartoffset = frameoffset
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+				rowheight = frame.height or frame:GetHeight() or 0
+				rowoffset = child.alignoffset or (rowheight / 2)
+				rowstartoffset = rowoffset
+			elseif child.width == "relative" then
+				safelayoutcall(child, "SetWidth", width * child.relWidth)
+
+				if child.DoLayout then
+					child:DoLayout()
+				end
+			elseif oversize then
+				if width > 1 then
+					frame:SetPoint("RIGHT", content)
+				end
+			end
+
+			if child.height == "fill" then
+				frame:SetPoint("BOTTOM", content)
+				isfullheight = true
+			end
+		end
+
+		--anchor the last row, if its full height needs a special case since  its height has just been changed by the anchor
+		if isfullheight then
+			rowstart:SetPoint("TOPLEFT", content, "TOPLEFT", 0, -height)
+		elseif rowstart then
+			rowstart:SetPoint("TOPLEFT", content, "TOPLEFT", 0, -(height + (rowoffset - rowstartoffset) + 3))
+		end
+
+		height = height + rowheight + 3
+		safecall(content.obj.LayoutFinished, content.obj, nil, height)
+	end)
diff --git a/Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.xml b/Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.xml
new file mode 100644
index 0000000..ae22c90
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/AceGUI-3.0.xml
@@ -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>
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua
new file mode 100644
index 0000000..200fc71
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-BlizOptionsGroup.lua
@@ -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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua
new file mode 100644
index 0000000..18e49cb
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-DropDownGroup.lua
@@ -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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua
new file mode 100644
index 0000000..bb93cc5
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Frame.lua
@@ -0,0 +1,311 @@
+--[[-----------------------------------------------------------------------------
+Frame Container
+-------------------------------------------------------------------------------]]
+local Type, Version = "Frame", 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, 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("gsTitleOptionExit")
+	frame.obj:Hide()
+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("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("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("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("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("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("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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua
new file mode 100644
index 0000000..1a330ef
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-InlineGroup.lua
@@ -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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua
new file mode 100644
index 0000000..1695f48
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-ScrollFrame.lua
@@ -0,0 +1,210 @@
+--[[-----------------------------------------------------------------------------
+ScrollFrame Container
+Plain container that scrolls its content and doesn't grow in height.
+-------------------------------------------------------------------------------]]
+local Type, Version = "ScrollFrame", 24
+local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
+if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
+
+local IsLegion = select(4, GetBuildInfo()) >= 70000
+
+-- Lua APIs
+local pairs, assert, type = pairs, assert, type
+local min, max, floor, abs = math.min, math.max, math.floor, math.abs
+
+-- 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 = 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
+		local curvalue = self.scrollbar:GetValue()
+		-- 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")
+				self:DoLayout()
+			end
+		else
+			if not self.scrollBarShown then
+				self.scrollBarShown = true
+				self.scrollbar:Show()
+				self.scrollframe:SetPoint("BOTTOMRIGHT", -20, 0)
+				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)
+		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
+	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)
+	if IsLegion then
+		scrollbg:SetColorTexture(0, 0, 0, 0.4)
+	else
+		scrollbg:SetTexture(0, 0, 0, 0.4)
+	end
+
+	--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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua
new file mode 100644
index 0000000..6e23abc
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-SimpleGroup.lua
@@ -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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua
new file mode 100644
index 0000000..ab63f31
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TabGroup.lua
@@ -0,0 +1,350 @@
+--[[-----------------------------------------------------------------------------
+TabGroup Container
+Container that uses tabs on top to switch between groups.
+-------------------------------------------------------------------------------]]
+local Type, Version = "TabGroup", 35
+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("igCharacterInfoTab")
+		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 status = self.status or self.localstatus
+		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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua
new file mode 100644
index 0000000..98a2080
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-TreeGroup.lua
@@ -0,0 +1,717 @@
+--[[-----------------------------------------------------------------------------
+TreeGroup Container
+Container that uses a tree control to switch between groups.
+-------------------------------------------------------------------------------]]
+local Type, Version = "TreeGroup", 40
+local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
+if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
+
+local IsLegion = select(4, GetBuildInfo()) >= 70000
+
+-- 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: GameTooltip, 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 frame = self.frame
+	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
+	local normalTexture = button:GetNormalTexture()
+	local line = button.line
+	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("Interface\\Buttons\\UI-PlusButton-UP")
+			toggle:SetPushedTexture("Interface\\Buttons\\UI-PlusButton-DOWN")
+		else
+			toggle:SetNormalTexture("Interface\\Buttons\\UI-MinusButton-UP")
+			toggle:SetPushedTexture("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()
+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
+	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
+		GameTooltip:SetOwner(frame, "ANCHOR_NONE")
+		GameTooltip:SetPoint("LEFT",frame,"RIGHT")
+		GameTooltip:SetText(frame.text:GetText() or "", 1, .82, 0, true)
+
+		GameTooltip:Show()
+	end
+end
+
+local function Button_OnLeave(frame)
+	local self = frame.obj
+	self:Fire("OnButtonLeave", frame.uniquevalue, frame)
+
+	if self.enabletooltips then
+		GameTooltip: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 frame = 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:SetPoint("TOPLEFT", frame, "TOPLEFT",0,0)
+	treeframe:SetPoint("BOTTOMLEFT", frame, "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
+		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
+		local hasChildren = self.hasChildren
+
+		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)
+		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
+
+		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)
+
+	if IsLegion then
+		scrollbg:SetColorTexture(0,0,0,0.4)
+	else
+		scrollbg:SetTexture(0,0,0,0.4)
+	end
+
+	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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua
new file mode 100644
index 0000000..33ed7d8
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIContainer-Window.lua
@@ -0,0 +1,331 @@
+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 = 4
+
+	local function frameOnClose(this)
+		this.obj:Fire("OnClose")
+	end
+
+	local function closeOnClick(this)
+		PlaySound("gsTitleOptionExit")
+		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("OnHide",frameOnClose)
+		frame:SetMinResize(240,240)
+		frame:SetToplevel(true)
+
+		local titlebg = frame:CreateTexture(nil, "BACKGROUND")
+		titlebg:SetTexture([[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([[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([[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([[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([[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([[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([[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([[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([[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([[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("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("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
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua
new file mode 100644
index 0000000..dbcad18
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Button.lua
@@ -0,0 +1,103 @@
+--[[-----------------------------------------------------------------------------
+Button Widget
+Graphical Button.
+-------------------------------------------------------------------------------]]
+local Type, Version = "Button", 23
+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("igMainMenuOption")
+	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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua
new file mode 100644
index 0000000..284ea8c
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-CheckBox.lua
@@ -0,0 +1,295 @@
+--[[-----------------------------------------------------------------------------
+Checkbox Widget
+-------------------------------------------------------------------------------]]
+local Type, Version = "CheckBox", 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 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("igMainMenuOptionCheckBoxOn")
+		else -- for both nil and false (tristate)
+			PlaySound("igMainMenuOptionCheckBoxOff")
+		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:GetHeight())
+			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(self.check, false)
+			self.check:Show()
+		else
+			--Nil is the unknown tristate value
+			if self.tristate and value == nil then
+				SetDesaturation(self.check, true)
+				self.check:Show()
+			else
+				SetDesaturation(self.check, false)
+				self.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("Interface\\Buttons\\UI-RadioButton")
+			checkbg:SetTexCoord(0, 0.25, 0, 1)
+			check:SetTexture("Interface\\Buttons\\UI-RadioButton")
+			check:SetTexCoord(0.25, 0.5, 0, 1)
+			check:SetBlendMode("ADD")
+			highlight:SetTexture("Interface\\Buttons\\UI-RadioButton")
+			highlight:SetTexCoord(0.5, 0.75, 0, 1)
+		else
+			size = 24
+			checkbg:SetTexture("Interface\\Buttons\\UI-CheckBox-Up")
+			checkbg:SetTexCoord(0, 1, 0, 1)
+			check:SetTexture("Interface\\Buttons\\UI-CheckBox-Check")
+			check:SetTexCoord(0, 1, 0, 1)
+			check:SetBlendMode("BLEND")
+			highlight:SetTexture("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: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:GetHeight())
+		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("Interface\\Buttons\\UI-CheckBox-Up")
+
+	local check = frame:CreateTexture(nil, "OVERLAY")
+	check:SetAllPoints(checkbg)
+	check:SetTexture("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("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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua
new file mode 100644
index 0000000..8e5f3ae
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-ColorPicker.lua
@@ -0,0 +1,194 @@
+--[[-----------------------------------------------------------------------------
+ColorPicker Widget
+-------------------------------------------------------------------------------]]
+local Type, Version = "ColorPicker", 23
+local AceGUI = LibStub and LibStub("AceGUI-3.0", true)
+if not AceGUI or (AceGUI:GetWidgetVersion(Type) or 0) >= Version then return end
+
+local IsLegion = select(4, GetBuildInfo()) >= 70000
+
+-- 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: ShowUIPanel, HideUIPanel, 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)
+	HideUIPanel(ColorPickerFrame)
+	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
+
+		ShowUIPanel(ColorPickerFrame)
+	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("Interface\\ChatFrame\\ChatFrameColorSwatch")
+	colorSwatch:SetPoint("LEFT")
+
+	local texture = frame:CreateTexture(nil, "BACKGROUND")
+	texture:SetWidth(16)
+	texture:SetHeight(16)
+	if IsLegion then
+		texture:SetColorTexture(1, 1, 1)
+	else
+		texture:SetTexture(1, 1, 1)
+	end
+	texture:SetPoint("CENTER", colorSwatch)
+	texture:Show()
+
+	local checkers = frame:CreateTexture(nil, "BACKGROUND")
+	checkers:SetWidth(14)
+	checkers:SetHeight(14)
+	checkers:SetTexture("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("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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua
new file mode 100644
index 0000000..9e5e4f5
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown-Items.lua
@@ -0,0 +1,477 @@
+--[[ $Id: AceGUIWidget-DropDown-Items.lua 1137 2016-05-15 10:57:36Z nevcairiel $ ]]--
+
+local AceGUI = LibStub("AceGUI-3.0")
+
+local IsLegion = select(4, GetBuildInfo()) >= 70000
+
+-- 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("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("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("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 = 3
+
+	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("igMainMenuOptionCheckBoxOn")
+		else
+			PlaySound("igMainMenuOptionCheckBoxOff")
+		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)
+		if IsLegion then
+			line:SetColorTexture(.5, .5, .5)
+		else
+			line:SetTexture(.5, .5, .5)
+		end
+		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
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua
new file mode 100644
index 0000000..d2b0f15
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-DropDown.lua
@@ -0,0 +1,737 @@
+--[[ $Id: AceGUIWidget-DropDown.lua 1116 2014-10-12 08:15:46Z nevcairiel $ ]]--
+local AceGUI = LibStub("AceGUI-3.0")
+
+-- Lua APIs
+local min, max, floor = math.min, math.max, math.floor
+local select, pairs, ipairs, type = select, pairs, ipairs, type
+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 = 30
+
+	--[[ 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("igMainMenuOptionCheckBoxOn") -- missleading name, but the Blizzard code uses this sound
+		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 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)
+
+			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
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua
new file mode 100644
index 0000000..6d16f37
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-EditBox.lua
@@ -0,0 +1,265 @@
+--[[-----------------------------------------------------------------------------
+EditBox Widget
+-------------------------------------------------------------------------------]]
+local Type, Version = "EditBox", 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 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("igMainMenuOptionCheckBoxOn")
+		HideButton(self)
+	end
+end
+
+local function EditBox_OnReceiveDrag(frame)
+	local self = frame.obj
+	local type, id, info = GetCursorInfo()
+	if type == "item" then
+		self:SetText(info)
+		self:Fire("OnEnterPressed", info)
+		ClearCursor()
+	elseif type == "spell" then
+		local name = GetSpellInfo(id, info)
+		self:SetText(name)
+		self:Fire("OnEnterPressed", name)
+		ClearCursor()
+	elseif type == "macro" then
+		local name = GetMacroInfo(id)
+		self:SetText(name)
+		self:Fire("OnEnterPressed", name)
+		ClearCursor()
+	end
+	HideButton(self)
+	AceGUI:ClearFocus()
+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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua
new file mode 100644
index 0000000..bf8b2c0
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Heading.lua
@@ -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("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("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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua
new file mode 100644
index 0000000..fdba650
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Icon.lua
@@ -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("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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua
new file mode 100644
index 0000000..e1388a1
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-InteractiveLabel.lua
@@ -0,0 +1,101 @@
+--[[-----------------------------------------------------------------------------
+InteractiveLabel Widget
+-------------------------------------------------------------------------------]]
+local Type, Version = "InteractiveLabel", 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 select, pairs = 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
+
+--[[-----------------------------------------------------------------------------
+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)
+
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
new file mode 100644
index 0000000..e77023a
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Keybinding.lua
@@ -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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua
new file mode 100644
index 0000000..9a181c5
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Label.lua
@@ -0,0 +1,166 @@
+--[[-----------------------------------------------------------------------------
+Label Widget
+Displays text and optionally an icon.
+-------------------------------------------------------------------------------]]
+local Type, Version = "Label", 23
+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:GetHeight()
+		else
+			-- image on the left
+			image:SetPoint("TOPLEFT")
+			if image:GetHeight() > label:GetHeight() 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:GetHeight())
+		end
+	else
+		-- no image shown
+		label:SetPoint("TOPLEFT")
+		label:SetWidth(width)
+		height = label:GetHeight()
+	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()
+
+		-- 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,
+}
+
+--[[-----------------------------------------------------------------------------
+Constructor
+-------------------------------------------------------------------------------]]
+local function Constructor()
+	local frame = CreateFrame("Frame", nil, UIParent)
+	frame:Hide()
+
+	local label = frame:CreateFontString(nil, "BACKGROUND", "GameFontHighlightSmall")
+	label:SetJustifyH("LEFT")
+	label:SetJustifyV("TOP")
+
+	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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
new file mode 100644
index 0000000..c9b0d86
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-MultiLineEditBox.lua
@@ -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)
diff --git a/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua
new file mode 100644
index 0000000..4a17767
--- /dev/null
+++ b/Titan/libs/Libs/AceGUI-3.0/widgets/AceGUIWidget-Slider.lua
@@ -0,0 +1,285 @@
+--[[-----------------------------------------------------------------------------
+Slider Widget
+Graphical Slider, like, for Range values.
+-------------------------------------------------------------------------------]]
+local Type, Version = "Slider", 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 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)
+	local self = frame.obj
+	if not frame.setup then
+		local newvalue = frame:GetValue()
+		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("igMainMenuOptionCheckBoxOn")
+		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)
diff --git a/Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.lua b/Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.lua
new file mode 100644
index 0000000..5a495f6
--- /dev/null
+++ b/Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.lua
@@ -0,0 +1,239 @@
+--[[ $Id: CallbackHandler-1.0.lua 3 2008-09-29 16:54:20Z nevcairiel $ ]]
+local MAJOR, MINOR = "CallbackHandler-1.0", 3
+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}
+
+local type = type
+local pcall = pcall
+local pairs = pairs
+local assert = assert
+local concat = table.concat
+local loadstring = loadstring
+local next = next
+local select = select
+local type = type
+local xpcall = xpcall
+
+local function errorhandler(err)
+	return geterrorhandler()(err)
+end
+
+local function CreateDispatcher(argCount)
+	local code = [[
+	local next, xpcall, eh = ...
+
+	local method, ARGS
+	local function call() method(ARGS) end
+
+	local function dispatch(handlers, ...)
+		local index
+		index, method = next(handlers)
+		if not method then return end
+		local OLD_ARGS = ARGS
+		ARGS = ...
+		repeat
+			xpcall(call, eh)
+			index, method = next(handlers, index)
+		until not method
+		ARGS = OLD_ARGS
+	end
+
+	return dispatch
+	]]
+
+	local ARGS, OLD_ARGS = {}, {}
+	for i = 1, argCount do ARGS[i], OLD_ARGS[i] = "arg"..i, "old_arg"..i end
+	code = code:gsub("OLD_ARGS", concat(OLD_ARGS, ", ")):gsub("ARGS", concat(ARGS, ", "))
+	return assert(loadstring(code, "safecall Dispatcher["..argCount.."]"))(next, xpcall, errorhandler)
+end
+
+local Dispatchers = setmetatable({}, {__index=function(self, argCount)
+	local dispatcher = CreateDispatcher(argCount)
+	rawset(self, argCount, dispatcher)
+	return dispatcher
+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, OnUsed, OnUnused)
+	-- TODO: Remove this after beta has gone out
+	assert(not OnUsed and not OnUnused, "ACE-80: OnUsed/OnUnused are deprecated. Callbacks are now done to registry.OnUsed and registry.OnUnused")
+
+	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
+
+		Dispatchers[select('#', ...) + 1](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"
+			if type(self)~="table" and type(self)~="string" then
+				error("Usage: "..RegisterName.."(self or \"addonId\", eventname, method): 'self or addonId': table or string 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.
+
diff --git a/Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.xml b/Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.xml
new file mode 100644
index 0000000..1aad3a2
--- /dev/null
+++ b/Titan/libs/Libs/CallbackHandler-1.0/CallbackHandler-1.0.xml
@@ -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>
\ No newline at end of file
diff --git a/Titan/libs/Libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua b/Titan/libs/Libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua
new file mode 100644
index 0000000..4a31cfe
--- /dev/null
+++ b/Titan/libs/Libs/LibSharedMedia-3.0/LibSharedMedia-3.0.lua
@@ -0,0 +1,292 @@
+--[[
+Name: LibSharedMedia-3.0
+Revision: $Revision: 91 $
+Author: Elkano (elkano@gmx.de)
+Inspired By: SurfaceLib by Haste/Otravi (troeks@gmail.com)
+Website: http://www.wowace.com/projects/libsharedmedia-3-0/
+Description: Shared handling of media data (fonts, sounds, textures, ...) between addons.
+Dependencies: LibStub, CallbackHandler-1.0
+License: LGPL v2.1
+]]
+
+local MAJOR, MINOR = "LibSharedMedia-3.0", 6010002 -- 6.1.0 v2 / increase manually on changes
+local lib = LibStub:NewLibrary(MAJOR, MINOR)
+
+if not lib then return end
+
+local _G = getfenv(0)
+
+local pairs		= _G.pairs
+local type		= _G.type
+
+local band			= _G.bit.band
+
+local table_insert	= _G.table.insert
+local table_sort	= _G.table.sort
+
+local locale = GetLocale()
+local locale_is_western
+local LOCALE_MASK = 0
+lib.LOCALE_BIT_koKR		= 1
+lib.LOCALE_BIT_ruRU		= 2
+lib.LOCALE_BIT_zhCN		= 4
+lib.LOCALE_BIT_zhTW		= 8
+lib.LOCALE_BIT_western	= 128
+
+local CallbackHandler = LibStub:GetLibrary("CallbackHandler-1.0")
+
+lib.callbacks		= lib.callbacks			or CallbackHandler:New(lib)
+
+lib.DefaultMedia	= lib.DefaultMedia		or {}
+lib.MediaList		= lib.MediaList			or {}
+lib.MediaTable		= lib.MediaTable		or {}
+lib.MediaType		= lib.MediaType			or {}
+lib.OverrideMedia	= lib.OverrideMedia		or {}
+
+local defaultMedia = lib.DefaultMedia
+local mediaList = lib.MediaList
+local mediaTable = lib.MediaTable
+local overrideMedia = lib.OverrideMedia
+
+
+-- create mediatype constants
+lib.MediaType.BACKGROUND	= "background"			-- background textures
+lib.MediaType.BORDER		= "border"				-- border textures
+lib.MediaType.FONT			= "font"				-- fonts
+lib.MediaType.STATUSBAR		= "statusbar"			-- statusbar textures
+lib.MediaType.SOUND			= "sound"				-- sound files
+
+-- populate lib with default Blizzard data
+-- BACKGROUND
+if not lib.MediaTable.background then lib.MediaTable.background = {} end
+lib.MediaTable.background["None"]									= [[]]
+lib.MediaTable.background["Blizzard Collections Background"]		= [[Interface\Collections\CollectionsBackgroundTile]]
+lib.MediaTable.background["Blizzard Dialog Background"]				= [[Interface\DialogFrame\UI-DialogBox-Background]]
+lib.MediaTable.background["Blizzard Dialog Background Dark"]		= [[Interface\DialogFrame\UI-DialogBox-Background-Dark]]
+lib.MediaTable.background["Blizzard Dialog Background Gold"]		= [[Interface\DialogFrame\UI-DialogBox-Gold-Background]]
+lib.MediaTable.background["Blizzard Garrison Background"]			= [[Interface\Garrison\GarrisonUIBackground]]
+lib.MediaTable.background["Blizzard Garrison Background 2"]			= [[Interface\Garrison\GarrisonUIBackground2]]
+lib.MediaTable.background["Blizzard Garrison Background 3"]			= [[Interface\Garrison\GarrisonMissionUIInfoBoxBackgroundTile]]
+lib.MediaTable.background["Blizzard Low Health"]					= [[Interface\FullScreenTextures\LowHealth]]
+lib.MediaTable.background["Blizzard Marble"]						= [[Interface\FrameGeneral\UI-Background-Marble]]
+lib.MediaTable.background["Blizzard Out of Control"]				= [[Interface\FullScreenTextures\OutOfControl]]
+lib.MediaTable.background["Blizzard Parchment"]						= [[Interface\AchievementFrame\UI-Achievement-Parchment-Horizontal]]
+lib.MediaTable.background["Blizzard Parchment 2"]					= [[Interface\AchievementFrame\UI-GuildAchievement-Parchment-Horizontal]]
+lib.MediaTable.background["Blizzard Rock"]							= [[Interface\FrameGeneral\UI-Background-Rock]]
+lib.MediaTable.background["Blizzard Tabard Background"]				= [[Interface\TabardFrame\TabardFrameBackground]]
+lib.MediaTable.background["Blizzard Tooltip"]						= [[Interface\Tooltips\UI-Tooltip-Background]]
+lib.MediaTable.background["Solid"]									= [[Interface\Buttons\WHITE8X8]]
+lib.DefaultMedia.background = "None"
+
+-- BORDER
+if not lib.MediaTable.border then lib.MediaTable.border = {} end
+lib.MediaTable.border["None"]								= [[]]
+lib.MediaTable.border["Blizzard Achievement Wood"]			= [[Interface\AchievementFrame\UI-Achievement-WoodBorder]]
+lib.MediaTable.border["Blizzard Chat Bubble"]				= [[Interface\Tooltips\ChatBubble-Backdrop]]
+lib.MediaTable.border["Blizzard Dialog"]					= [[Interface\DialogFrame\UI-DialogBox-Border]]
+lib.MediaTable.border["Blizzard Dialog Gold"]				= [[Interface\DialogFrame\UI-DialogBox-Gold-Border]]
+lib.MediaTable.border["Blizzard Party"]						= [[Interface\CHARACTERFRAME\UI-Party-Border]]
+lib.MediaTable.border["Blizzard Tooltip"]					= [[Interface\Tooltips\UI-Tooltip-Border]]
+lib.DefaultMedia.border = "None"
+
+-- FONT
+if not lib.MediaTable.font then lib.MediaTable.font = {} end
+local SML_MT_font = lib.MediaTable.font
+--[[
+All font files are currently in all clients, the following table depicts which font supports which charset as of 5.0.4
+Fonts were checked using langcover.pl from DejaVu fonts (http://sourceforge.net/projects/dejavu/) and FontForge (http://fontforge.org/)
+latin means check for: de, en, es, fr, it, pt
+
+file				name							latin	koKR	ruRU	zhCN	zhTW
+2002.ttf			2002							X		X		X		-		-
+2002B.ttf			2002 Bold						X		X		X		-		-
+ARHei.ttf			AR CrystalzcuheiGBK Demibold	X		-		X		X		X
+ARIALN.TTF			Arial Narrow					X		-		X		-		-
+ARKai_C.ttf			AR ZhongkaiGBK Medium (Combat)	X		-		X		X		X
+ARKai_T.ttf			AR ZhongkaiGBK Medium			X		-		X		X		X
+bHEI00M.ttf			AR Heiti2 Medium B5				-		-		-		-		X
+bHEI01B.ttf			AR Heiti2 Bold B5				-		-		-		-		X
+bKAI00M.ttf			AR Kaiti Medium B5				-		-		-		-		X
+bLEI00D.ttf			AR Leisu Demi B5				-		-		-		-		X
+FRIZQT__.TTF		Friz Quadrata TT				X		-		-		-		-
+FRIZQT___CYR.TTF	FrizQuadrataCTT					x		-		X		-		-
+K_Damage.TTF		YDIWingsM						-		X		X		-		-
+K_Pagetext.TTF		MoK								X		X		X		-		-
+MORPHEUS.TTF		Morpheus						X		-		-		-		-
+MORPHEUS_CYR.TTF	Morpheus						X		-		X		-		-
+NIM_____.ttf		Nimrod MT						X		-		X		-		-
+SKURRI.TTF			Skurri							X		-		-		-		-
+SKURRI_CYR.TTF		Skurri							X		-		X		-		-
+
+WARNING: Although FRIZQT___CYR is available on western clients, it doesn't support special European characters e.g. é, ï, ö
+Due to this, we cannot use it as a replacement for FRIZQT__.TTF
+]]
+
+if locale == "koKR" then
+	LOCALE_MASK = lib.LOCALE_BIT_koKR
+--
+	SML_MT_font["굵은 글꼴"]		= [[Fonts\2002B.TTF]]
+	SML_MT_font["기본 글꼴"]		= [[Fonts\2002.TTF]]
+	SML_MT_font["데미지 글꼴"]		= [[Fonts\K_Damage.TTF]]
+	SML_MT_font["퀘스트 글꼴"]		= [[Fonts\K_Pagetext.TTF]]
+--
+	lib.DefaultMedia["font"] = "기본 글꼴" -- someone from koKR please adjust if needed
+--
+elseif locale == "zhCN" then
+	LOCALE_MASK = lib.LOCALE_BIT_zhCN
+--
+	SML_MT_font["伤害数字"]		= [[Fonts\ARKai_C.ttf]]
+	SML_MT_font["默认"]			= [[Fonts\ARKai_T.ttf]]
+	SML_MT_font["聊天"]			= [[Fonts\ARHei.ttf]]
+--
+	lib.DefaultMedia["font"] = "默认" -- someone from zhCN please adjust if needed
+--
+elseif locale == "zhTW" then
+	LOCALE_MASK = lib.LOCALE_BIT_zhTW
+--
+	SML_MT_font["提示訊息"]		= [[Fonts\bHEI00M.ttf]]
+	SML_MT_font["聊天"]			= [[Fonts\bHEI01B.ttf]]
+	SML_MT_font["傷害數字"]		= [[Fonts\bKAI00M.ttf]]
+	SML_MT_font["預設"]			= [[Fonts\bLEI00D.ttf]]
+--
+	lib.DefaultMedia["font"] = "預設" -- someone from zhTW please adjust if needed
+
+elseif locale == "ruRU" then
+	LOCALE_MASK = lib.LOCALE_BIT_ruRU
+--
+	SML_MT_font["2002"]								= [[Fonts\2002.TTF]]
+	SML_MT_font["2002 Bold"]						= [[Fonts\2002B.TTF]]
+	SML_MT_font["AR CrystalzcuheiGBK Demibold"]		= [[Fonts\ARHei.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium (Combat)"]	= [[Fonts\ARKai_C.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium"]			= [[Fonts\ARKai_T.TTF]]
+	SML_MT_font["Arial Narrow"]						= [[Fonts\ARIALN.TTF]]
+	SML_MT_font["Friz Quadrata TT"]					= [[Fonts\FRIZQT___CYR.TTF]]
+	SML_MT_font["MoK"]								= [[Fonts\K_Pagetext.TTF]]
+	SML_MT_font["Morpheus"]							= [[Fonts\MORPHEUS_CYR.TTF]]
+	SML_MT_font["Nimrod MT"]						= [[Fonts\NIM_____.ttf]]
+	SML_MT_font["Skurri"]							= [[Fonts\SKURRI_CYR.TTF]]
+--
+	lib.DefaultMedia.font = "Friz Quadrata TT"
+--
+else
+	LOCALE_MASK = lib.LOCALE_BIT_western
+	locale_is_western = true
+--
+	SML_MT_font["2002"]								= [[Fonts\2002.TTF]]
+	SML_MT_font["2002 Bold"]						= [[Fonts\2002B.TTF]]
+	SML_MT_font["AR CrystalzcuheiGBK Demibold"]		= [[Fonts\ARHei.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium (Combat)"]	= [[Fonts\ARKai_C.TTF]]
+	SML_MT_font["AR ZhongkaiGBK Medium"]			= [[Fonts\ARKai_T.TTF]]
+	SML_MT_font["Arial Narrow"]						= [[Fonts\ARIALN.TTF]]
+	SML_MT_font["Friz Quadrata TT"]					= [[Fonts\FRIZQT__.TTF]]
+	SML_MT_font["MoK"]								= [[Fonts\K_Pagetext.TTF]]
+	SML_MT_font["Morpheus"]							= [[Fonts\MORPHEUS_CYR.TTF]]
+	SML_MT_font["Nimrod MT"]						= [[Fonts\NIM_____.ttf]]
+	SML_MT_font["Skurri"]							= [[Fonts\SKURRI_CYR.TTF]]
+--
+	lib.DefaultMedia.font = "Friz Quadrata TT"
+--
+end
+
+-- STATUSBAR
+if not lib.MediaTable.statusbar then lib.MediaTable.statusbar = {} end
+lib.MediaTable.statusbar["Blizzard"]						= [[Interface\TargetingFrame\UI-StatusBar]]
+lib.MediaTable.statusbar["Blizzard Character Skills Bar"]	= [[Interface\PaperDollInfoFrame\UI-Character-Skills-Bar]]
+lib.MediaTable.statusbar["Blizzard Raid Bar"]				= [[Interface\RaidFrame\Raid-Bar-Hp-Fill]]
+lib.DefaultMedia.statusbar = "Blizzard"
+
+-- SOUND
+if not lib.MediaTable.sound then lib.MediaTable.sound = {} end
+lib.MediaTable.sound["None"]								= [[Interface\Quiet.ogg]]	-- Relies on the fact that PlaySound[File] doesn't error on non-existing input.
+lib.DefaultMedia.sound = "None"
+
+local function rebuildMediaList(mediatype)
+	local mtable = mediaTable[mediatype]
+	if not mtable then return end
+	if not mediaList[mediatype] then mediaList[mediatype] = {} end
+	local mlist = mediaList[mediatype]
+	-- list can only get larger, so simply overwrite it
+	local i = 0
+	for k in pairs(mtable) do
+		i = i + 1
+		mlist[i] = k
+	end
+	table_sort(mlist)
+end
+
+function lib:Register(mediatype, key, data, langmask)
+	if type(mediatype) ~= "string" then
+		error(MAJOR..":Register(mediatype, key, data, langmask) - mediatype must be string, got "..type(mediatype))
+	end
+	if type(key) ~= "string" then
+		error(MAJOR..":Register(mediatype, key, data, langmask) - key must be string, got "..type(key))
+	end
+	mediatype = mediatype:lower()
+	if mediatype == lib.MediaType.FONT and ((langmask and band(langmask, LOCALE_MASK) == 0) or not (langmask or locale_is_western)) then return false end
+	if mediatype == lib.MediaType.SOUND and type(data) == "string" then
+		local path = data:lower()
+		-- Only ogg and mp3 are valid sounds.
+		if not path:find(".ogg", nil, true) and not path:find(".mp3", nil, true) then
+			return false
+		end
+	end
+	if not mediaTable[mediatype] then mediaTable[mediatype] = {} end
+	local mtable = mediaTable[mediatype]
+	if mtable[key] then return false end
+
+	mtable[key] = data
+	rebuildMediaList(mediatype)
+	self.callbacks:Fire("LibSharedMedia_Registered", mediatype, key)
+	return true
+end
+
+function lib:Fetch(mediatype, key, noDefault)
+	local mtt = mediaTable[mediatype]
+	local overridekey = overrideMedia[mediatype]
+	local result = mtt and ((overridekey and mtt[overridekey] or mtt[key]) or (not noDefault and defaultMedia[mediatype] and mtt[defaultMedia[mediatype]])) or nil
+	return result ~= "" and result or nil
+end
+
+function lib:IsValid(mediatype, key)
+	return mediaTable[mediatype] and (not key or mediaTable[mediatype][key]) and true or false
+end
+
+function lib:HashTable(mediatype)
+	return mediaTable[mediatype]
+end
+
+function lib:List(mediatype)
+	if not mediaTable[mediatype] then
+		return nil
+	end
+	if not mediaList[mediatype] then
+		rebuildMediaList(mediatype)
+	end
+	return mediaList[mediatype]
+end
+
+function lib:GetGlobal(mediatype)
+	return overrideMedia[mediatype]
+end
+
+function lib:SetGlobal(mediatype, key)
+	if not mediaTable[mediatype] then
+		return false
+	end
+	overrideMedia[mediatype] = (key and mediaTable[mediatype][key]) and key or nil
+	self.callbacks:Fire("LibSharedMedia_SetGlobal", mediatype, overrideMedia[mediatype])
+	return true
+end
+
+function lib:GetDefault(mediatype)
+	return defaultMedia[mediatype]
+end
+
+function lib:SetDefault(mediatype, key)
+	if mediaTable[mediatype] and mediaTable[mediatype][key] and not defaultMedia[mediatype] then
+		defaultMedia[mediatype] = key
+		return true
+	else
+		return false
+	end
+end
diff --git a/Titan/libs/Libs/LibSharedMedia-3.0/lib.xml b/Titan/libs/Libs/LibSharedMedia-3.0/lib.xml
new file mode 100644
index 0000000..5b4a5c8
--- /dev/null
+++ b/Titan/libs/Libs/LibSharedMedia-3.0/lib.xml
@@ -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="LibSharedMedia-3.0.lua" />
+</Ui>
\ No newline at end of file
diff --git a/Titan/libs/Libs/LibStub/LibStub.lua b/Titan/libs/Libs/LibStub/LibStub.lua
new file mode 100644
index 0000000..cfc97de
--- /dev/null
+++ b/Titan/libs/Libs/LibStub/LibStub.lua
@@ -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
diff --git a/Titan/libs/Libs/LibStub/LibStub.toc b/Titan/libs/Libs/LibStub/LibStub.toc
new file mode 100644
index 0000000..b43e20c
--- /dev/null
+++ b/Titan/libs/Libs/LibStub/LibStub.toc
@@ -0,0 +1,13 @@
+## 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
+## X-Curse-Packaged-Version: 1.0
+## X-Curse-Project-Name: LibStub
+## X-Curse-Project-ID: libstub
+## X-Curse-Repository-ID: wow/libstub/mainline
+
+LibStub.lua
diff --git a/Titan/libs/changelog.txt b/Titan/libs/changelog.txt
new file mode 100755
index 0000000..65a2e06
--- /dev/null
+++ b/Titan/libs/changelog.txt
@@ -0,0 +1,401 @@
+Ace3 Release - Revision r1182 (July 17th, 2018)
+-----------------------------------------------
+- AceComm-3.0: Support numeric targets for custom channels (Ticket #419)
+- AceComm-3.0/ChatThrottleLib: Updated for WoW 8.0 API changes
+- AceConfig-3.0: Allow number values for the "width" parameter, which acts as a multiplier of the default width
+- AceConfigDialog-3.0: Enable iconCoords for the main level of a tree group (Ticket #417)
+- AceGUI-3.0: Implement a Table Layout (#442)
+- AceGUI-3.0: EditBox: Only try to handle dragging if a valid object is being dragged into the text box
+- AceGUI-3.0: Various fixes and work-arounds for WoW 8.0 changes
+
+Ace3 Release - Revision r1166 (August 29th, 2017)
+-------------------------------------------------
+- Updated any use of PlaySound to the new numeric constants (Fix for WoW 7.3)
+- AceConfigDialog-3.0: implement fallback validation feedback in a StaticPopup (Ticket #2)
+
+Ace3 Release - Revision r1158 (March 28th, 2017)
+------------------------------------------------
+- AceGUI-3.0: Fire OnShow events from Frame and Windows containers (Ticket #396)
+- AceGUI-3.0: Add SetJustifyV/H API to the Label/InteractiveLabel widgets (Ticket #397)
+
+Ace3 Release - Revision r1151 (July 18th, 2016)
+-----------------------------------------------
+- AceConfig-3.0: Allow specifying images using the numeric fileID (Ticket #389)
+- AceGUI-3.0: Use SetColorTexture in WoW 7.0
+- AceGUI-3.0: Expose the HighlightText API for EditBox and MultiLineEditBox Widgets (Ticket #378)
+- AceGUI-3.0: Keybinding: Support MouseWheel Up/Down bindings (Ticket #372)
+- AceGUI-3.0: TreeGroup: Refresh the tree in an OnUpdate once after Acquire, not only after creation (Fixes missing tree in WoW 7.0)
+- AceSerializer-3.0: Create consistent tokens for infinity, independent of the clients platform
+
+Ace3 Release - Revision r1134 (June 23rd, 2015)
+-----------------------------------------------
+- AceGUI-3.0: TreeGroup: Prevent Word Wrap in tree elements
+
+Ace3 Release - Revision r1128 (February 24th, 2015)
+---------------------------------------------------
+- AceGUI-3.0: Fixed an error in GameTooltip handling causing wrong formatting in some cases
+
+Ace3 Release - Revision r1122 (October 14th, 2014)
+--------------------------------------------------
+- AceDB-3.0: Now using the GetCurrentRegion() API to determine region-based profile keys (fix for WoW 6.0)
+- AceComm-3.0: Update ChatThrottleLib to v23
+- AceConfigDialog-3.0: Show a more meaningful title on Blizzard Options Groups (Ticket #353)
+- AceGUI-3.0: ColorPicker: Frame Level is automatically increased to ensure the color picker frame is above the option panel
+- AceGUI-3.0: DropDown: Properly disable all clickable areas when the dropdown is disabled (Ticket #360)
+- AceHook-3.0: Always use HookScript when hooking scripts in a secure fashion (Ticket #338)
+- AceTimer-3.0: New timer implementation based on C_Timer.After
+
+Ace3 Release - Revision r1109 (February 19th, 2014)
+---------------------------------------------------
+- AceComm-3.0: Ambiguate addon comm messages to restore behavior to be identical to pre-5.4.7
+- AceConfigRegistry-3.0: Added an option to skip options table validation on registering
+
+Ace3 Release - Revision r1104 (October 31st, 2013)
+--------------------------------------------------
+- AceGUI-3.0: Flow Layout: Added a safeguard to prevent an infinite loop in the layout engine
+- AceGUI-3.0: DropDown: Adjust its style slightly to closer resemble the Blizzard DropDown Widget
+- AceGUI-3.0: DropDown: API enhancements to specify the width of the pullout and be notified when its opened
+
+Ace3 Release - Revision r1098 (September 13th, 2013)
+----------------------------------------------------
+- AceDB-3.0: Switch characters to the default profile if their active profile is deleted (Ticket #324)
+- AceConfigDialog-3.0: Try to prevent static popup taint (Ticket #322)
+- AceGUI-3.0: Button: Add a new "Auto Width" option (Ticket #310)
+- AceGUI-3.0: DropDown: Make the entire DropDown widget clickable (Ticket #339)
+- AceGUI-3.0: EditBox: Allow dragging macros to the editbox (which will then contain the macros name) (Ticket #337)
+- AceGUI-3.0: Slider: Add a workaround for the broken slider steps in WoW 5.4 (Ticket #346)
+- AceGUI-3.0: TreeGroup: Fix an issue introduced by 5.4 broken scrollbars (Ticket #345)
+- AceHook-3.0: Allow hooking of AnimationGroup scripts (Ticket #314)
+
+Ace3 Release - Revision r1086 (May 21st, 2013)
+----------------------------------------------
+- AceAddon-3.0: Improved behavior when loading modules during game startup, ensures proper loading order
+
+Ace3 Release - Revision r1083 (March 4th, 2013)
+-----------------------------------------------
+- AceTimer-3.0: Fixed an issue that caused the parameter list passed to the callback to be cut off at nil values
+- AceGUI-3.0: InlineGroup: The title attribute is properly reset for recycled objects
+
+Ace3 Release - Revision r1078 (February 10th, 2013)
+---------------------------------------------------
+- AceTimer-3.0: Re-write based on AnimationTimers
+- AceHook-3.0: Improved checks/error messages when trying to hook a script on a "nil" frame
+- AceDBOptions-3.0: Added Italian locale
+- AceGUI-3.0: BlizOptionsGroup: Fixed the "default" button callback
+- AceGUI-3.0: Colorpicker: The colorpicker is now clamped to the screen
+
+Ace3 Release - Revision r1061 (August 27th, 2012)
+-------------------------------------------------
+- AceConfigDialog-3.0: Try to avoid potential taints in static popup dialogs
+- AceConfigDialog-3.0: Sort multiselects with "radio" style
+- AceGUI-3.0: Support for WoW 5.0
+- AceGUI-3.0: MultiLineEditBox: Support shift-click'ing items/spells/etc. into the editbox
+- AceGUI-3.0: Label: Fix text alignment (Ticket #301)
+- AceGUI-3.0: Checkbox: Description text on a disable checkbox should look disabled (Ticket #304)
+- AceGUI-3.0: Keybinding: Ensure the Keybinding popup is on the top level (Ticket #305)
+
+Ace3 Release - Revision r1041 (November 29th, 2011)
+---------------------------------------------------
+- AceDB-3.0: Added locale and factionrealmregion profile keys
+- AceSerializer-3.0: Removed support for NaN, as WoW 4.3 does no longer allow it.
+- AceGUI-3.0: Frame: Add :EnableResize (Ticket #214)
+
+Ace3 Release - Revision r1032 (June 29th, 2011)
+-----------------------------------------------
+- AceTab-3.0: Improvements to Match handling (Ticket #255 and #256)
+- AceGUI-3.0: DropDown layout fix with hidden labels (Ticket #234)
+
+Ace3 Release - Revision r1025 (April 27th, 2011)
+------------------------------------------------
+- AceComm-3.0: Updated for 4.1 changes - now handles RegisterAddonMessagePrefix internally for you.
+- AceGUI-3.0: TabGroup: Fixed width of tabs in 4.1(Ticket #243)
+
+Ace3 Release - Revision r1009 (February 9th, 2011)
+--------------------------------------------------
+- AceLocale-3.0: Fix erronous assumption that the default locale is always the first to be registered for the :NewLocale() "silent" flag. The flag must now be set on the FIRST locale to be registered.
+- AceLocale-3.0: The :NewLocale() "silent" flag may now be set to the string "raw", meaning nils are returned for unknown translations.
+- AceGUI-3.0: Fix the disabled state of Icon widgets
+- AceGUI-3.0: The header of the Frame widget now dynamically changes size to fit the text (Ticket #171)
+- AceGUI-3.0: Its now possible to define a custom order the elements in a dropdown widget
+- AceGUI-3.0: Improved widget focus behaviour across the board (Ticket #192, #193)
+- AceGUI-3.0: Fixed a bug that made it impossible to block the tree widget from being user resizable (Ticket #163)
+- AceGUI-3.0: Fixed a bug that caused TreeGroups to become unresponsive under certain conditions (Ticket #189, #202)
+- AceGUI-3.0: Enhanced the DropDown widget to allow it to be reused more easily.
+- AceConfigDialog-3.0: Select Groups now have the proper order in the dropdown (Ticket #184)
+- AceConfigDialog-3.0: Implemented "radio" style select boxes (Ticket #149)
+
+Ace3 Release - Revision r981 (October 27th, 2010)
+-------------------------------------------------
+- AceAddon-3.0: Fixed a library upgrading issue
+- AceAddon-3.0: Modules are now enabled in loading order
+- AceGUI-3.0: Keybinding: The widget will no longer steal keybindings even when inactive (Ticket #169)
+- AceGUI-3.0: EditBox: Fixed spell drag'n'drop
+
+
+Ace3 Release - Revision r971 (October 12th, 2010)
+-------------------------------------------------
+- Small fixes and adjustments for the 4.0 Content Patch.
+- AceGUI-3.0: ScrollFrame: Allow for a small margin of error when determining if the scroll bar should be shown.
+- AceGUI-3.0: Added new widget APIs: GetText for EditBox and DisableButton for MultiLineEditBox
+
+Ace3 Release - Revision r960 (July 20th, 2010)
+----------------------------------------------
+- AceGUI-3.0: Label: Reset Image Size and TexCoords on Acquire (Ticket #110)
+- AceGUI-3.0: CheckBox: Re-apply the disabled state after setting a value, so the visuals are correct in either case. (Ticket #107)
+- AceGUI-3.0: Icon: Fix the vertical size. It'll now properly scale with the image size, and not be fixed to about 110px. (Ticket #104)
+- AceGUI-3.0: External Containers (Frame, Window) should always start in a visible state. (Ticket #121)
+- AceGUI-3.0: Added Blizzard sounds to widgets (Ticket #120)
+- AceGUI-3.0: CheckBox: check for self.desc:GetText() being nil as well as "" to prevent setting the wrong height on the checkbox causing bouncing checkboxes.
+- AceGUI-3.0: Rewrite of the MultiLineEditBox (Ticket #68)
+- AceGUI-3.0: CheckBox: Fix alignment of the text in OnMouseDown when an image is set. (Ticket #142)
+- AceGUI-3.0: Add SetMaxLetters APIs to EB and MLEB (Ticket #135)
+- AceGUI-3.0: Frame: Add events for OnEnter/OnLeave of the statusbar (Ticket #139)
+- AceGUI-3.0: Major cleanups and refactoring in nearly all widgets and containers.
+- AceConfigDialog-3.0: Always obey the min/max values on range-type widgets (Ticket #114)
+- AceConfigDialog-3.0: Pass iconCoords set on groups in the options table to the tree widget (Ticket #111)
+- AceConfigDialog-3.0: Implement "softMin" and "softMax", allowing for a UI-specific minimum/maximum for range controls, while allowing manual input of values in the old min/max range. (Ticket #123)
+- AceConfigDialog-3.0: Don't close frames in CloseAll that are being opened after the CloseAll event was dispatched. (Ticket #132).
+- AceSerializer-3.0: Fix encoding & decoding of \030. (Ticket #115)
+- AceDB-3.0: Remove empty sections on logout, keeping the SV clean of more useless informations.
+- AceDBOptions-3.0.lua: Fix a string typo (Ticket #141)
+
+Ace3 Release - Revision r907 (December 16th, 2009)
+---------------------------------------------------
+- AceGUI-3.0: Frame: Properly save the width in the status table.
+- AceConfigCmd-3.0: Properly handle help output of inline groups with a different handler. (Ticket #101)
+- AceConfigDialog-3.0: Don't bail out and error when a dialogControl was invalid, instead show the error and fallback to the default control for that type.
+- AceConfigDialog-3.0: Fix a hickup with the OnUpdate script not getting upgraded properly in some situations.
+
+Ace3 Release - Revision r900 (December 8th, 2009)
+--------------------------------------------------
+- AceGUI-3.0: Alot of visual fixes regarding margins and general widget styles.
+- AceGUI-3.0: Ability to accept links for EditBox Widget (Ticket #21)
+- AceGUI-3.0: ScrollFrame: Hide the scrollbar when there is no overflowing content, and allow the Layout functions to use that space for widgets.
+- AceGUI-3.0: DropDown: Added a GetValue() API to the Widget (Ticket #69)
+- AceGUI-3.0: Button: Pass the arguments of the OnClick handler to the OnClick callback (Ticket #57)
+- AceGUI-3.0: add a Window container, basically a plain window with close button
+- AceGUI-3.0: Add support for inline descriptions to the checkbox widget.
+- AceGUI-3.0: Added an API to the Window container to disable the user-resizing of the same. (Ticket #80)
+- AceGUI-3.0: TreeGroup: Allow iconCoords to be passed for the tree elements. (Ticket #59)
+- AceGUI-3.0: Slider: Add a more visible backdrop/border around the manual input area (Ticket #98, #46)
+- AceGUI-3.0: Allow displaying a image in front of the checkbox label. (Ticket #82)
+- AceConfig-3.0: Added an experimental "descStyle" member to all option table nodes that allows you to control the way the description is presented.
+                 Supported values are "tooltip" for the old behaviour, and "inline" for a inline display of the description, pending support in AceGUI-3.0 widgets.
+- AceConfigCmd-3.0: Properly parse functions and methods supplied for the "hidden" option table member. (Ticket #96)
+- AceConfigDialog-3.0: Fix the unpacking of the basepath arguments when internally calling :Open (Ticket #90)
+- AceConfigDialog-3.0: Properly refresh BlizOptions Windows which are registered with a path on NotifyChange. (Ticket #93)
+- AceConfigDialog-3.0: Allow image/imageCoords on toogle elements (Note that the width/height of the image on the toggle cannot be changed) (Ticket #82)
+- AceConfigDialog-3.0: Pass the groups "name" tag to DropDownGroups as the title. (Ticket #79)
+- AceDB-3.0: Remove the metatable from the DB before removing defaults, so we don't accidentally invoke it in the process. (Ticket #66)
+- AceDB-3.0: Don't save the profileKeys for namespaces, since we use the profile of the parent DB anyway. This will cut down on SV complexity when using alot of namespaces.
+- AceDB-3.0: Don't fire the OnProfileReset callback when copying a profile.
+- AceDBOptions-3.0: Show the current profile on the dialog. (Ticket #56)
+- AceComm-3.0: Add callbacks for message chunks going out the wire (via CTL). Useful for displaying progress for very large messages.
+- AceConsole-3.0: Add :Printf() so you don't have to do Print(format())
+
+Ace3 Beta - Revision 820 (August 7th, 2009)
+--------------------------------------------
+- AceComm-3.0: Updated ChatThrottleLib to v21
+- AceGUI-3.0: Fixed a glitch in the TabGroup code that caused tabs to be unresponsive under rare conditions. (Ticket #38)
+- AceGUI-3.0: Consistent "disabled" behaviour of all widgets. (Ticket #47)
+- AceGUI-3.0: Changed the way widgets are handled on release to avoid a crash in the game client. (Ticket #49)
+- AceGUI-3.0: Fixed a glitch in the button graphics. (Ticket #58)
+- AceGUI-3.0: Localized the "Close" Text on the Frame widget.
+
+Ace3 Beta - Revision 803 (April 14th, 2009)
+--------------------------------------------
+- AceConfig-3.0: Allow spaces in the keys of config tables. Spaces will be changed on the fly to underscores in AceConfigCmd-3.0 - there is no collision check in place, yet.
+- AceConfig-3.0: Support a "fontSize" attribute to the description type. Possible values are "small" (default), "medium" and "large".
+- AceConfigDialog-3.0: Fixed an error that would occur when calling InterfaceOptionsFrame_OpenToCategory from within an event handler in a Blizzard Options integrated frame. (Ticket #33)
+- AceConfigDialog-3.0: The "execute" type does now recognize the "image" attributes, and will display a clickable icon instead of the button when an image is supplied. (Ticket #35)
+- AceConfigDialog-3.0: Pass icons defined in the option table to the TreeGroup widget (Ticket #20)
+- AceConfigDialog-3.0: Fixed a bug that caused an empty group widget to be drawn if all groups were hidden.
+- AceConfigCmd-3.0: Improved the behaviour of select and multiselect elements. (Ticket #26)
+- AceDB-3.0: Add a GetNamespace function to the DB Objects which returns an existing namespace from the DB object.
+- AceGUI-3.0 Slider Widget: Properly show percentage values as min/max if isPercent is true. (Ticket #32)
+- AceGUI-3.0: Fixed an error in the TreeGroup Widget that caused execution to stop if no name was provided.
+- AceGUI-3.0: Fixed the behaviour of the MultiLineEditbox Widget (Accept button not clickable). (Ticket #28)
+- AceGUI-3.0: TabGroup: Set a maximum width for tabs based on the size of the widget. (Ticket #34)
+- AceGUI-3.0: Added a new InteractiveLabel with OnEnter/OnLeave/OnClick callbacks and a highlight texture
+- AceGUI-3.0: Add SetFont and SetFontObject functions to the Label widget (and the new InteractiveLabel)
+- AceGUI-3.0: Support icons in the TreeGroup display. (Ticket #20)
+- AceGUI-3.0: Added a new :SetRelativeWidth Widget-API that allows you to set the width of widgets relative to their container.
+- AceGUI-3.0: Alot of fixes, tweaks and consistency changes.
+
+Ace3 Beta - Revision 741 (Feb 15th, 2009)
+--------------------------------------------
+- AceDBOptions-3.0: Disable the "Copy From" and "Delete" dropdowns if there are no profiles to choose from. (Ticket #19)
+- AceGUI-3.0: Improve TabGroup visual style - only stretch them to the full width if they would use more then 75% of the exisiting space.
+- AceGUI-3.0: Added a third optional argument to <container>:AddChild() to specify the position for the new widget. (Ticket #22)
+- AceConfigCmd-3.0: Improve help output when viewing groups.
+- AceConfigDialog-3.0: Refresh the Options Panel after a confirmation is canceled to reset the value to its previous value. (Ticket #23)
+- AceDB-3.0: Fix a data inconsistency when using false as a table key. (Ticket #25)
+
+Ace3 Beta - Revision 722 (Jan 4th, 2009)
+--------------------------------------------
+- AceHook-3.0: Fix :SecureHookScript to not fail on previously empty scripts since frame:HookScript does nothing at all in that case. (Ticket #16)
+- AceLocale-3.0: Implement 'silent' option for :NewLocale to disable the warnings on unknown entrys (Ticket #18)
+- AceTimer-3.0: Implement :TimeLeft(handle) function (Ticket #10)
+- AceGUI-3.0: Fix TabGroup tab resizing to be consistent
+- AceGUI-3.0: Fixed EditBox alignment when the label is disabled (Ticket #13)
+- AceDB-3.0: Implement OnProfileShutdown callback (Ticket #7)
+- AceDBOptions-3.0: Updated esES and ruRU locale
+
+Ace3 Beta - Revision 706 (Oct 18th, 2008)
+--------------------------------------------
+- First Beta release after WoWAce move
+- Removed WoW 2.4.x compat layer
+- AceGUI-3.0: Fix disabling of the Multiline Editbox
+- AceGUI-3.0: Improvements to the Keybinding Widget
+
+Ace3 Beta - Revision 81437 (Sept 6th, 2008)
+--------------------------------------------
+- AceConfigDialog-3.0: the confirm callback will now receive the new value that is being set (same signature as the validate callback)
+- AceConfigDialog-3.0: :Close and :CloseAll are now safe to call from within callbacks.
+- AceGUI-3.0: Added new methods to the widget base table, see ACE-205 for full reference
+- AceGUI-3.0: Various fixes to Widgets and recycling process
+- Now compatible with WoW 3.0 (compat layer is to be removed upon 3.0 release)
+
+
+Ace3 Beta - Revision 76325 (June 9th, 2008)
+--------------------------------------------
+- AceGUI-3.0: Finish Multiselect support for the Dropdown widget (nargiddley)
+- AceGUI-3.0: Re-write TabGroup layouting (nargiddley)
+- AceGUI-3.0: TreeGroup: Add :EnableButtonTooltips(enable) to make the default tooltips on the tree optional, enabled by default. (nargiddley)
+- AceGUI-3.0: TabGroup: Add OnTabEnter and OnTabLeave Callbacks  (nargiddley)
+- AceConfigDialog-3.0: Add :SelectGroup(appName, ...) - Selects the group given by the path specified then refreshes open windows. (nargiddley)
+- AceConfigDialog-3.0: :Open now accepts an optional path, when given will open the window with only the given group and its children visible (nargiddley)
+- AceConfigDialog-3.0: :AddToBlizOptions now accepts an optional path, this will add the config page to display the specified group and its children only. (nargiddley)
+- AceConfigDialog-3.0: ACE-189: allow multiselect to be shown as a dropdown by setting dialogControl = "Dropdown" (nargiddley)
+- AceConfigDialog-3.0: Add Custom tooltips to the TreeGroup and TabGroup, shows both name and desc for the group. (nargiddley)
+- AceConfigCmd-3.0: ACE-195: Remove unneeded references to .confirm, will no longer error when .confirm is a boolean (nargiddley)
+- AceAddon-3.0: Allow for an optional first argument to NewAddon to be a table to be used as the base for the addon. (ammo)
+
+Ace3 Beta - Revision 74633 (May 19th, 2008)
+--------------------------------------------
+- AceTimer-3.0: ACE-173: don't error on nil handle for CancelTimer(), just bail out early. (ammo)
+- AceGUI-3.0: ACE-161, ACE-180, ACE-181: New and improved DropDown widget (originally coded by Borlox) (nargiddley,nevcairiel)
+- AceGUI-3.0: AceGUI will call OnWidthSet and OnHeightSet as frames resize (nargiddley)
+- AceGUI-3.0: TabGroup: Use OptionsFrameTabButtonTemplate for tabs (nargiddley)
+- AceGUI-3.0: TabGroup: Tabs now span multiple lines when there are too many to fit in the width of the frame (nargiddley)
+- AceGUI-3.0: TreeGroup: Tree is now sizable by dragging, orig patch by yssaril (nargiddley)
+- AceGUI-3.0: Flow layout will now reduce widgets width to fit rather than leaving them sticking out the side of container widgets (nargiddley)
+- AceGUI-3.0: Dropdowns will no longer be left open in the background when the frame is clicked or other widgets are activated (nargiddley)
+- AceGUI-3.0: ACE-159: Rename Release to OnRelease and Acquire to OnAcquire for widgets. (nargiddley)
+- AceGUI-3.0: ACE-171: add IsVisible and IsShown methods to the widget metatable (nargiddley)
+- AceGUI-3.0: ACE-164: add tooltips to tree to show full text of childs that got clipped (ammo)
+- AceGUI-3.0: ACE-174: make buttons in AceGUI-3.0 locale independant (ammo)
+- AceGUI-3.0: ACE-166: fix treegroup visual bug (ammo)
+- AceGUI-3.0: ACE-184: make numeric entry for slider more intuitive (ammo)
+- AceConfigCmd-3.0: ACE-172 - ignore description in cmd (ammo)
+- AceConsole-3.0:  nolonger check for existance of slashcommands, overwrite where needed. Last one wins, this enables AddonLoader to X-LoadOn-Slash and override the slashcommand from AddonLoader slashcommand with an Ace3 one. (Ammo)
+
+Ace3 Beta - Revision 69509 (April 13th, 2008)
+---------------------------------------------
+- AceComm-3.0: turn off error messages when receiving invalid multi-part messages (its happening on login etc) (nevcairiel)
+- AceDBOptions-3.0: shorten info text at top to prevent scrollbars. (nevcairiel)
+- AceHook-3.0: ACE-162: fix unhooking of objects that were not actually hooked (nevcairiel)
+- AceDB-3.0: fire the DB callbacks after the namespaces changed their profile as well (nevcairiel)
+- AceDB-3.0: namespaces can now be individually reset using :ResetProfile() on the namespace directly (nevcairiel)
+- AceDB-3.0: added a optional argument to :ResetProfile to not populate the reset to all namespaces (so the main profile can reset individually without reseting all namespaces too)  (nevcairiel)
+
+Ace3 Beta - Revision 66329 (March 27th, 2008)
+---------------------------------------------
+- Overall 2.4 clean ups - removing 2.4 checks and work arounds (nevcairiel)
+- AceBucket-3.0: clear the timer reference when unregistering a bucket to prevent a error when unregistering a bucket that was never fired (nevcairiel)
+- AceAddon-3.0: Bugfix when enabling/disabling modules from the parents OnEnable after disabling / enabling the parent addon. (ammo)
+- AceGUI-3.0: Don't parent the BlizOptionsGroup widget to UIParent and Hide it by default. Fixes stray controls on the screen. (nargiddley)
+- AceConfigDialog-3.0: Config windows without a default size won't incorrectly get a default size from a previously open window. (nargiddley)
+- AceDBOptions-3.0: added zhCN and zhTW locale (nevcairiel)
+
+Ace3 Beta - Revision 65665 (March 25th, 2008)
+---------------------------------------------
+- AceGUI-3.0: ACE-139: Changed all Widgets to resemble the Blizzard 2.4 Options Style (nevcairiel)
+- AceGUI-3.0: Fixed "List"-Layout not reporting new width to "fill"-mode widgets (mikk)
+- AceGUI-3.0: added :SetColor to the Label widget (nevcairiel)
+- AceGUI-3.0: ACE-132: ColorPicker: added checkers texture for better alpha channel display, and fixed "white"-texture bug (nevcairiel,nargiddley,ammo)
+- AceConfig-3.0: ACE-113: Added uiName, uiType, handler, option, type to the info table (nevcairiel,nargiddley)
+- AceConfigDialog-3.0: ACE-139: Adjusted for 2.4 options panels (nevcairiel)
+- AceConfigDialog-3.0: Use "width" parameter for the description widget (if present) (nevcairiel)
+- AceConfigDialog-3.0: ACE-135: Add support for specifying a rowcount for multiline editboxes (nargiddley)
+- AceConfigDialog-3.0: :AddToBlizOptions will return the frame registered so you can use it in InterfaceOptionsFrame_OpenToFrame (nevcairiel)
+- AceConfigCmd-3.0: handle "hidden" in help-output (nevcairiel)
+- AceHook-3.0: fix unhooking of secure hooks (nevcairiel)
+- AceDBOptions-3.0: add optional argument to GetOptionsTable(db[, noDefaultProfiles]) - if set to true will not show the default profiles in the profile selection (nevcairiel)
+- AceDBOptions-3.0: added koKR locale (nevcairiel)
+- Ace3 Standalone: Removed the "Ace3" Category from the 2.4 options panel (nevcairiel)
+
+Ace3 Beta - Revision 64176 (March 10th, 2008)
+---------------------------------------------
+- AceGUI-3.0: Improve Alpha handling for the ColorPicker widget, ColorPicker widget closes the ColorPickerFrame before opening to prevent values getting carried over (nargiddley)
+- AceGUI-3.0: The Slider widget will only react to the mousewheel after it has been clicked (anywhere including the label) to prevent accidental changes to the value when trying to scroll the container it is in (nargiddley)
+- AceGUI-3.0: The TreeGroup widget is scrollable with the mousewheel (nargiddley)
+- AceGUI-3.0: ACE-154: Fix frame levels in more cases to prevent widgets ending up behind their containers (nargiddley)
+- AceConfigDialog: Color picker obeys hasAlpha on the color type (nargiddley)
+- AceConfigDialog-3.0: ACE-155: Make sure that the selected group is type='group' when checking if it exists (nargiddley)
+- AceDBOptions-3.0: added frFR locale (nevcairiel)
+
+Ace3 Beta - Revision 63886 (March 8th, 2008)
+---------------------------------------------
+- AceDBOptions-3.0: new library to provide a Ace3Options table to control the AceDB-3.0 profiles (nevcairiel)
+- AceDB-3.0: add "silent" option to DeleteProfile and CopyProfile when we deal with namespaces (nevcairiel)
+- AceDB-3.0: implement library upgrade path (nevcairiel)
+- AceDB-3.0: ACE-146: fix problem with non-table values overruling ['*']-type defaults (nevcairiel)
+- AceConsole-3.0: treat |T|t texture links similar to |H|h|h links. (ammo)
+- AceGUI-3.0: Use Blizzard Templates for the EditBox and DropDown widget (nevcairiel)
+- AceBucket-3.0: ACE-150: callback is now optional, if not supplied will use the eventname as method name (only possible if one event is supplied, and not a event table) (nevcairiel)
+- tests: adjust tests for AceGUI and AceConsole changes (nevcairiel)
+
+Ace3 Beta - Revision 63220 (Feb 29th, 2008)
+---------------------------------------------
+- AceTimer-3.0: CancelAllTimers() now cancels silent (elkano)
+- AceConfigDialog: Add :SetDefaultSize(appName, width, height), sets the size the dialog will open to. Does not effect already open windows.  (nargiddley)
+- AceConfigDialog: Fix typo in type check for range values (nargiddley)
+- AceGUI: ColorPicker widget will correctly fire OnValueChanged for the cancel event of the colorpicker popup.  Reset ColorPicker's color on Acquire.  (nargiddley)
+- AceGUI: Fix Spelling of Aquire -> Acquire for widgets, all custom widgets will need to be updated.  A warning will be printed for widgets not upgraded yet.  (nargiddley)
+- AceConfigCmd-3.0: add simple coloring to slashcommand output. (ammo)
+- AceConsole-3.0: add some color to :Print (ammo)
+- AceAddon-3.0: set error level on library embedding to point to the :NewAddon call (nevcairiel)
+
+Ace3 Beta - Revision 62182 (Feb 20th, 2008)
+---------------------------------------------
+- Ace3 StandAlone: Add a page to the Blizzard 2.4 Interface Options with icons to open dialogs for configs registered when installed standalone (nargiddley)
+- AceConfigDialog: type = 'description' now uses the fields image and imageCoords instead of icon and iconCoords, add imageWidth and imageHeight (nargiddley)
+- AceConfigDialog: Add :AddToBlizzardOptions(appName, name), this will add the specified config to the Blizzard Options pane new in 2.4.  This will only be available if running on the 2.4 PTR (nargiddley)
+- AceDB: fix GetProfiles() when setting the same profile twice (nevcairiel)
+- AceDB: bail out of :SetProfile early when trying to set to the same profile (nevcairiel)
+- AceDB: add nil checks to metatable handling (nevcairiel)
+- AceDB: clear tables that are empty after defaults removal (nevcairiel)
+- AceGUI: Fix a couple of layout bugs causing the width of groups to be wrong (nargiddley)
+- AceGUI: Add Icon widget (nargiddley)
+- AceGUI: Allow room for the border in the BlizOptionsGroup widget (nargiddley)
+- AceGUI: Button and Keybinding use UIPanelButtonTemplate2 (nargiddley)
+- AceConsole-3.0: Fix bug where no table for [self] was created when registering weak commands (ammo)
+- AceTimer-3.0: add missing :OnEmbedDisable (ammo)
+- AceAddon-3.0: added :GetName() that will always return the "real" name of a addon or module object without any prefixes (nevcairiel)
+
+Ace3 Beta - Revision 60697 (Feb 9th, 2008)
+---------------------------------------------
+- CallbackHandler-1.0: remove unnecessary table creation if a event is fired thats not registered (nevcairiel)
+- AceAddon-3.0: fixed a bug with recursive addon loading (nevcairiel)
+- AceGUI: Update TabGroup's tablist format, tabs are selected by value not index (nargiddley)
+- AceGUI: Add MultiLineEditBox widget (nargiddley, originally by bam)
+- AceGUI: Small fix to the flow layout preventing controls overlapping in some cases (nargiddley)
+- AceConfigDialog: Implement control and dialogControl for types 'input' and 'select' (nargiddley)
+- AceConfigDialog: Add support for multiline = true on type = 'input' (nargiddley)
+- AceConfigDialog: Fix an error when all groups are hidden in a group with childGroups = 'select' (nargiddley)
+- AceConfigDialog: type = 'description' will now show .icon as an image with its text (nargiddley)
+- AceConfigDialog: multiline inputs are no longer forced to width = "full" (nargiddley)
+- AceConfigDialog: bug fix when loading without AceConsole present (nevcairiel)
+
+Ace3 Beta - Revision 60545 (Feb 7th, 2008)
+---------------------------------------------
+- AceGUI: SetToplevel(true) for the Frame widget, multiple open windows should play nice together now (nargiddley)
+- AceGUI: Move Frames to the FULLSCREEN_DIALOG strata (nargiddley)
+- AceGUI: Dropdown, Editbox and Keybinding labels grey out when disabled (nargiddley)
+- AceGUI: Add OnClick callback to the TreeGroup widget (nargiddley)
+- AceConfigDialog: Confirm popups will be above the config window (nargiddley)
+
+Ace3 Beta - Revision 60163 (Feb 3rd, 2008)
+---------------------------------------------
+- Initial Beta release
\ No newline at end of file
diff --git a/Titan/libs/lib.xml b/Titan/libs/lib.xml
new file mode 100755
index 0000000..fd1c0cd
--- /dev/null
+++ b/Titan/libs/lib.xml
@@ -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">
+	<Include file="LibSharedMedia-3.0\lib.xml"/>
+</Ui>
+
diff --git a/Titan/libs/widget.xml b/Titan/libs/widget.xml
new file mode 100644
index 0000000..7e4a4c7
--- /dev/null
+++ b/Titan/libs/widget.xml
@@ -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">
+	<Include file="AceGUI-3.0-SharedMediaWidgets\widget.xml" />
+</Ui>
\ No newline at end of file
diff --git a/Titan/license.txt b/Titan/license.txt
new file mode 100644
index 0000000..b9d7fbe
--- /dev/null
+++ b/Titan/license.txt
@@ -0,0 +1,34 @@
+Copyright (c) 2018, Titan Panel Development Team
+
+All rights reserved.
+
+Redistribution without modification, is permitted under the following conditions:
+
+    * Redistributions of source code, must retain the above copyright notice,
+      this list of conditions and the following disclaimer.
+    * Redistributions in binary form must reproduce the above copyright notice,
+      this list of conditions and the following disclaimer in the documentation
+      and/or other materials provided with the distribution.
+    * Redistribution or providing means for redistribution of a release package
+      (in compressed form or otherwise) is prohibited without prior written
+      authorization from a representative of the Titan Panel Development Team.
+      Release versions (packages) hosted on officially endorsed websites are
+      excluded from this clause.  The Titan Panel Development Team reserves the
+      right, at its sole discretion, to authorize additional websites to host
+      the Software at their location and provide redistribution services. In
+      addition, the Titan Panel Development Team reserves the right to cease
+      Software updates without prior notice or ask for removal of versions of
+      the Software, from specific Internet locations that were previously
+      authorized to host the Software.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
+CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
\ No newline at end of file
diff --git a/Titan/locale/Localization.BR.lua b/Titan/locale/Localization.BR.lua
new file mode 100644
index 0000000..c81800c
--- /dev/null
+++ b/Titan/locale/Localization.BR.lua
@@ -0,0 +1,556 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","ptBR")
+if not L then return end
+
+L["TITAN_PANEL"] = "Painel Titan";
+local TITAN_PANEL = "Painel Titan";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/A";
+L["TITAN_SECONDS"] = "segundos";
+L["TITAN_MINUTES"] = "minutos";
+L["TITAN_HOURS"] = "horas";
+L["TITAN_DAYS"] = "dias";
+L["TITAN_SECONDS_ABBR"] = "s";
+L["TITAN_MINUTES_ABBR"] = "min";
+L["TITAN_HOURS_ABBR"] = "h";
+L["TITAN_DAYS_ABBR"] = "d";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "kB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kbps"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "Nenhum";
+L["TITAN_USE_COMMA"] = "Use vírgula";
+L["TITAN_USE_PERIOD"] = "Use ponto";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "Você não pode apagar seu perfil atual.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."Atenção : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "Se você deseja continuar com esta operação, aperte 'Aceitar' (sua IU irá recarregar), de outra forma aperte 'Cancelar'ou a tecla 'Esc'."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."Esta configuração irá resetar sua(s) barra(s) e configurações do "..TITAN_PANEL.." para valores padrão e irá recriar seu perfil. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."This will reload "..TITAN_PANEL..". "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = TITAN_PANEL.." Tentativas"
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Tentativas"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "O plugin abaixo requisitou ser registrado com o "..TITAN_PANEL..".\n"
+	.. "Por favor mande quaisquer problemas para o autor do plugin."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Tipo"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Categoria"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Nome do Botão"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Status"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Problema"
+L["TITAN_PANEL_EXTRAS"] = "Extras do " .. TITAN_PANEL
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Notas"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Índice de tabela"
+L["TITAN_PANEL_EXTRAS_SHORT"] = "Extras"
+L["TITAN_PANEL_EXTRAS_DESC"] = "Esss plugins não estão com dados de configuração carregados no momento.\n"
+	.. "Esses são seguros para excluir."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "Apagar dados de configuração"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "Entrada de configuração foi removida."
+L["TITAN_PANEL_CHARS"] = "Personagens"
+L["TITAN_PANEL_CHARS_DESC"] = "Estes são os personagens com dados de configuração."
+L["TITAN_PANEL_REGISTER_START"] = "Registrando plugins do " .. TITAN_PANEL;
+L["TITAN_PANEL_REGISTER_END"] = "Processo de registro feito."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {reset | reseta tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffffReseta o "..TITAN_PANEL.." para os valores/posições padrão.";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffReseta a escala de fonte de dica do "..TITAN_PANEL.." para o padrão.";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffReseta a transparência de dica do "..TITAN_PANEL.." para o padrão.";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffReseta a escala do "..TITAN_PANEL.." para o padrão.";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffReseta o espaçamento do botão do "..TITAN_PANEL.." para o padrão.";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffffAbre a janela de controle do "..TITAN_PANEL.."";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffffAbre a janela de controle de transparência";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffffAbre a janela de controle de Pele";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <name> <server>: |cffffffffConfigura o perfil para o perfil salvo requisitado.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<name>: |cffffffffpode ser tanto o nome do personagem quanto o nome do perfil personalizado."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<server>: |cffffffffpode ser tanto o nome do servidor quanto 'TitanCustomProfile'."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffAlterna o "..TITAN_PANEL.." para carregar silencioamente.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." escala de fonte de dica resetada.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." transparência de dica resetada.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." escala resetada.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." espaçamento de botão resetado.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Global";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Perfil Global";
+L["TITAN_PANEL_GLOBAL_USE"] = "Usar Perfil Global";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Usar como Perfil Global";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Usar um perfil global para todos os personagens";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "resetando opções";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "Você não pode carregar um perfil quando um perfil global está em uso";
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffffd700 pelo Time de Desenvolvimento do |cffff8c00"..TITAN_PANEL;
+L["TITAN_PANEL_MENU_TITLE"] = TITAN_PANEL;
+L["TITAN_PANEL_MENU_HIDE"] = "Ocultar ";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(Em Combate)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(Recarregar IU)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "Exibir Texto Colorido";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "Exibir Ícone";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "Exibir Texto de Rótulo";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "Ocultar Automaticamente";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "Centralizar Texto";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Exibir Barra";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "Desativar Ajuste de Tela";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "Desativar Ajuste de Mini-mapa";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "Ajuste Automático de Log";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "Ajuste Automático de Mochila";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "Ajuste Automático do Quadro de Ticket";
+L["TITAN_PANEL_MENU_PROFILES"] = "Perfis";
+L["TITAN_PANEL_MENU_PROFILE"] = "Perfil ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "Personalizado";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " foi apagado.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "Reino";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "Personagem";
+
+
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "Sua UI será recarregada quando o botão 'Ok' for apertado, permitindo que seu perfil personalizado seja salvo.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "Digite um nome para seu perfil personalizado:\n(máximo 20 caracteres, espaços não são permitidos, maíusculas e minúsculas)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "Configurações atuais serão salvas no perfil de nome: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "O nome de perfil digitado já existe. Você tem certeza que quer sobrescrevê-lo? Aperte 'Aceitar' se sim, de outra maneira aperte 'Cancelar' ou a tecla 'Esc'.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "Gerenciar";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "Carregar";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "Apagar";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "Salvar";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "Configuração";
+L["TITAN_PANEL_OPTIONS"] = "Opções";
+L["TITAN_PANEL_MENU_TOP"] = "Superior"
+L["TITAN_PANEL_MENU_TOP2"] = "Superior 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "Inferior"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "Interior 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = "Dicas e Quadros do " .. TITAN_PANEL .."";
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Dicas e Quadros";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Barras Superiores"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Barras Inferiores"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "Barras"
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = "Barras Superiores do " .. TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = "Barras Inferiores do " .. TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Dicas";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "Quadros";
+L["TITAN_PANEL_MENU_PLUGINS"] = "Plugins";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "Trancar Botões";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "Exibir Versões dos Plugins";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "Plugin no Lado Direito";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "Força os LDB Launchers para o Lado Direito";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"Incorporados","Geral","Combate","Informação","Interface","Profissão"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "Exibir Dicas";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "Ocultar Dicas em Combate";
+
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Trancar auto-Ocultar barras quando em combate";
+L["TITAN_PANEL_MENU_RESET"] = "Resetar o " .. TITAN_PANEL .. " para os Padrões";
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "Peles";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "Painel de Fontes"
+L["TITAN_PANEL_MENU_ENABLED"] = "Ativado";
+L["TITAN_PANEL_MENU_DISABLED"] = "Desativado";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Mover a Esquerda";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Mover a Direita";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Exibir Texto do Plugin";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Sempre Ativado";
+L["TITAN_PANEL_MENU_POSITION"] = "Posição";
+L["TITAN_PANEL_MENU_BAR"] = "Barra";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Escolha em qual barra o plugin é exibido";
+L["TITAN_PANEL_MENU_SHOW"] = "Exibir Plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Atualizar Plugins";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Atualizar Texto e Posição do Plugin";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Carregament Silencioso";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Versão";
+L["TITAN_ABOUT_AUTHOR"] = "Autor";
+L["TITAN_ABOUT_CREDITS"] = "Créditos";
+L["TITAN_ABOUT_CATEGORY"] = "Categoria";
+L["TITAN_ABOUT_EMAIL"] = "E-mail";
+L["TITAN_ABOUT_WEB"] = "Website";
+L["TITAN_ABOUT_LICENSE"] = "Licença";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "Addon de barra de exibição de informações. Permite que usuários adicionem alimentação de dados ou lançadores de plugins num painel de controle colocado acima ou abaixo da tela.";
+L["TITAN_TRANS_MENU_TEXT"] = "Transparência do " .. TITAN_PANEL;
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "Transparência";
+L["TITAN_TRANS_MENU_DESC"] = "Ajustar transparência para barras e dicas do "..TITAN_PANEL.." ";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "Barra Principal";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "Barra Auxiliar";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "Dica";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "Configura a transparência para as dicas de diversos plugins.";
+L["TITAN_UISCALE_MENU_TEXT"] = "Escala e Fonte do " .. TITAN_PANEL;
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Escala e Fonte";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "Escala da IU";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "Escala do " .. TITAN_PANEL;
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "Espaçamento de Botões";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Espaçamento de Ícones";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "Escala da Fonte de Dica";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "Desativar Escala da Fonte de Dica";
+L["TITAN_UISCALE_MENU_DESC"] = "Controla vários aspectos da IU e "..TITAN_PANEL..".";
+L["TITAN_UISCALE_SLIDER_DESC"] = "Define a escala de toda a sua IU.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "Define a escala para os diversos botões e ícones do "..TITAN_PANEL.." ";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "Adjusts the space between left-side plugins.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Adjusts the space between right-side plugins.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "Adjusts the scale for the tooltip of the various plugins.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "Disables "..TITAN_PANEL.." Tooltip Font Scale Control.";
+
+L["TITAN_SKINS_TITLE"] = "Skins " .. TITAN_PANEL;
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Skins - Personalizado";
+L["TITAN_SKINS_TITLE_CUSTOM"] = "Skins Personalizadas do " .. TITAN_PANEL;
+L["TITAN_SKINS_MAIN_DESC"] = "All custom skins are assumed to be in: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ ".."\n"
+			.."\n"..TITAN_PANEL.." and custom skins are stored under the Custom folder."
+L["TITAN_SKINS_LIST_TITLE"] = "Skin List";
+L["TITAN_SKINS_SET_DESC"] = "Select a skin for the "..TITAN_PANEL.." bars.";
+L["TITAN_SKINS_SET_HEADER"] = "Set "..TITAN_PANEL.." Skin";
+L["TITAN_SKINS_RESET_HEADER"] = "Reset "..TITAN_PANEL.." Skins";
+L["TITAN_SKINS_NEW_HEADER"] = "Add New Skin";
+L["TITAN_SKINS_NAME_TITLE"] = "Skin Name"
+L["TITAN_SKINS_NAME_DESC"] = "Enter a name for your new skin. It will be used in the skin dropdown lists.";
+L["TITAN_SKINS_PATH_TITLE"] = "<Skin Folder>"
+L["TITAN_SKINS_PATH_DESC"] = "<Skin Folder> under the "..TITAN_PANEL.." install. See the example above."
+L["TITAN_SKINS_ADD_HEADER"] = "Add Skin";
+L["TITAN_SKINS_ADD_DESC"] = "Adds a new skin to the list of available skins for "..TITAN_PANEL..".";
+L["TITAN_SKINS_REMOVE_HEADER"] = "Remove Skin";
+L["TITAN_SKINS_REMOVE_DESC"] = "Select a custom skin to remove."
+L["TITAN_SKINS_REMOVE_BUTTON"] = "Remove";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "Removes the selected custom skin.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "You are responsible for removing any unwanted custom skins "
+	.."from the "..TITAN_PANEL.." install folder. Addons can not add or remove files."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "Reset to Defaults";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "Resets the skin list to the default "..TITAN_PANEL.." skins.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "Select the font type for the various plugins on the "..TITAN_PANEL.." Bars.";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "Font Size";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "Sets the size for the "..TITAN_PANEL.." font.";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = ""..TITAN_PANEL.." Frame Strata";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "Adjusts the frame strata for the "..TITAN_PANEL.." Bar(s).";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Avançado";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Change Timers only if you experience issues with frames not adjusting.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Entering World";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving world or an instance.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Vehicle";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving vehicle.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "Toggles " .. TITAN_PANEL .. " auto-Ocultar on/off feature";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "Bolsas: ";
+L["TITAN_BAG_TOOLTIP"] = "Informações de Bolsas";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "Dica: Clique para abrir todas as bolsas.";
+L["TITAN_BAG_MENU_TEXT"] = "Bolsa";
+L["TITAN_BAG_USED_SLOTS"] = "Espaços Usados";
+L["TITAN_BAG_FREE_SLOTS"] = "Espaços Vazios";
+L["TITAN_BAG_BACKPACK"] = "Mochila";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "Exibir Espaços Usados";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "Exibir Espaços Disponíveis";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "Exibir Tooltip Detalhada";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "Ignorar Contêineres";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "Ignorar Bolsas de Profissão";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Saco de Magitrama Encantado", "Bolsa de Runatrama Encantada", "Algibeira do Encantador", "Grande Bolsa de Encantamentos", "Bolsa de Fogo Místico",
+"Bolsa Misteriosa", "Bolsa Sobrenatural", "Bolsa Tarde Encantadora - Exclusividade \"Lepos'Tiche\""};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Caixa de Ferramentas Pesada", "Caixa de Ferramentas de Ferrovil", "Caixa de Ferramentas de Titânico", "Caixa de Ferramentas de Elemêntio", "Bolsa de Alta Tecnologia - Linha \"Lepos'Tiche - Maddy\""};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Bolsa de Herborismo", "Bolsa de Herborismo Cenariana", "Algibeira de Cenarius", "Sacola Botânica de Mika", "Bolsa Esmeralda", "Bolsa de Expedição Hyjal",
+"Bolsa de Carga de Ervas - Linha \"Lepos'Tiche Esverdeada\""};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Algibeira do Escriba", "Pacote de Bolsos Infinitos", "Mochila Escolar - Linha \"Lepos'Tiche - Xandera\"", "Algibeira do Escriba Real", "Burnished Inscription Bag"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Bolsa de Gemas", "Saco de Joias", "Amarra Cravejada de Gemas - Exclusividade \"Lepos'Tiche\"", "Bolsa de Gemas de Seda Luxuosa"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Algibeira do Coureiro", "Bolsa de Muitos Pelegos", "Mala de Viagem do Coureador", "Bolsa de Couro \"Lepos'Tiche - Miya\"", "Burnished Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Saco de Mineração", "Bolsa de Mineração Reforçada", "Bolsa de Mineração de Mamute", "Bolsa de Metal Precioso \"Lepos'Tiche - Christina\"", "Bolsa de Mineração Triplamente Reforçada", "Burnished Mining Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Caixa de Pesca do Mestre Anzol"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Portable Refrigerator", "Advanced Refrigeration Unit"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "Relógio";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "Server Offset Hour Value: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "Horário Local: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "Horário do Servidor: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "Horário Ajustado do Servidor: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "Hint: Left-click to adjust the offset hour"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "(server time only) and the 12/24H time format.";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Shift Left-Click to toggle the Calendar on/off.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "Server Hour Offset: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "Offset";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24H";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "Altera a exbição do horário entre os formatos AM/PM ou 24 horas.";
+L["TITAN_CLOCK_MENU_TEXT"] = "Relógio";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "Exibir Horário Local (L)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "Exibir Horário do Servidor (S)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "Exibir Horário Ajustado do Servidor (A)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "Exibir no Lado Direito";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "Ocultar Botão de Hora/Calendário";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Ocultar Botão de Hora";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Ocultar Botão do Calendário";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "Formato das Coordenadas";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "Loc: ";
+L["TITAN_LOCATION_TOOLTIP"] = "Informações de Localização";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "Dica: Shift + Clique para adicionar informações"
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "de sua localização para o chat message.";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "Zona: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "Sub Zona: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "Informação JvJ: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "Localização da Casa";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "Estalagem: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "Localização";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "Exibir Zone Text";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "Exibir Coordinates on World Map";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "Cursor: %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "Jogador: %s";
+L["TITAN_LOCATION_NO_COORDS"] = "Sem Coordenadas";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "Exibir Location Name Above Minimap";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Update World Map When Zone Changes";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "QPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "QPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "QPS Atual: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "QPS Médio: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "QPS Mínimo: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "QPS Máximo: ";
+L["TITAN_FPS_TOOLTIP"] = "Quadros por Segundo";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "Latência: ";
+L["TITAN_LATENCY_TOOLTIP"] = "Status da Rede";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Latência do Reino (local): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Latência do Jogo (global): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "Banda de Entrada: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "Banda de Saída: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Latência";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "Saque: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "\"Cada Um Por Si\"";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "\"Rodízio\"";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "Mestre Saqueador";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "Saque em Grupo";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "Necessidade sobre Ganância";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "Informação de Tipo de Saque";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "Tipo de Saque";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "Jogada Aleatória";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "Dica: Clique para uma jogada aleatória.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "Selecione o tipo de jogada com o menu acessível com o botão direito do mouse.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "Dificuldade da Masmorra";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "Dificuldade da Raide";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "Exibir Dificuldade da Masmorra/Raide";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "Configurar Dificuldade da Masmorra";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "Configurar Dificuldade da Raide";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "Automático (Baseado no Grupo)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "Memória: ";
+L["TITAN_MEMORY_TOOLTIP"] = "Uso da Memória";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "Atual: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "Inicial: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "Taxa de Aumento: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "Informações de Performance";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "Performance";
+L["TITAN_PERFORMANCE_ADDONS"] = "Addon Usage";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "Uso de Memória de Addons";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "Addon Memory Format";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "Uso de CPU por Addons";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "Nome:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "Uso";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "Aumento";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "Total de Memória de Addon:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "Tempo de CPU Total:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "Exibir QPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "Exibir Latência do Reino";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Exibir Latência do Jogo";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "Exibir Memória";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "Exibir Addon Memory Usage";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "Exibir Addon Usage Rate";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "CPU Profiling Mode";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "Enable CPU Profiling Mode ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "Disable CPU Profiling Mode ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "Addons Monitorados: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "Addons Monitorados";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "Dica: Clique para forçar a coleção de lixo.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "EXP/hr This Level: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "EXP/hr This Session: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "Time To Level: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "Level Complete: ";
+L["TITAN_XP_TOTAL_RESTED"] = "Rested: ";
+L["TITAN_XP_XPTOLEVELUP"] = "EXP To Level: ";
+L["TITAN_XP_TOOLTIP"] = "EXP Info";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "Total Time Played: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "Time Played This Level: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "Time Played This Session: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "Total XP Required This Level: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "XP Gained This Level: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "XP Needed To Level: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "XP Gained This Session: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "XP/HR This Level: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "XP/HR This Session: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "Time To Level (Level Rate): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "Time To Level (Session Rate): ";
+L["TITAN_XP_MENU_TEXT"] = "EXP";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "Exibir XP/HR This Level";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "Exibir XP/HR This Session";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Exibir Multi-Info View";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "Button";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "Exibir Rested XP";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "Exibir XP To Level";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "Exibir Estimated Kills To Level";
+L["TITAN_XP_MENU_RESET_SESSION"] = "Resetar Sessão";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "Refresh Timers";
+L["TITAN_XP_UPDATE_PENDING"] = "Atualizando...";
+L["TITAN_XP_KILLS_LABEL"] = "Kills To Level (at %s XP gained last): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "Est. Kills: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "Session Time: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "Exibir Session Time";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*) dies, you gain (%d+) experience.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "Est. Gains: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "XP Gains To Level (at %s XP gained last): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "Exibir Estimated XP Gains To Level";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "Conserto",
+	tooltip = "Informações de Conserto",
+	button = "Durabilidade: ",
+	normal = "Custo de Conserto (Normal): ",
+	friendly = "Custo de Conserto (Respeitado): ",
+	honored = "Custo de Conserto (Honrado): ",
+	revered = "Custo de Conserto (Reverenciado): ",
+	exalted = "Custo de Conserto (Exaltado): ",
+	buttonNormal = "Exibir Normal",
+	buttonFriendly = "Exibir Respeitado (5%)",
+	buttonHonored = "Exibir Honrado (10%)",
+	buttonRevered = "Exibir Reverenciado (15%)",
+	buttonExalted = "Exibir Exaltado (20%)",
+	percentage = "Exibir como Porcentagem",
+	itemnames = "Exibir Nome dos Itens",
+	mostdamaged = "Exibir o Mais Danificado",
+	Exibirdurabilityframe = "Exibir Quadro de Durabilidade",
+	undamaged = "Exibir Itens Não Danificados",
+	discount = "Desconto",
+	nothing = "Nada Danificado",
+	confirmation = "Do you want to repair all items ?",
+	badmerchant = "This merchant cannot repair. Displaying normal repair costs instead.",
+	popup = "Exibir Popup de Conserto",
+	showinventory = "Calculate Inventory Damage",
+	WholeScanInProgress = "Atualizando...",
+	AutoReplabel = "Auto Conserto",
+	AutoRepitemlabel = "Auto Consertar Todos os Itens",
+	ShowRepairCost = "Exibir Custo de Conserto",
+	ignoreThrown = "Ignore Thrown",
+	ShowItems = "Exibir Itens",
+	ShowDiscounts = "Exibir Descontos",
+	ShowCosts = "Exibir Custos",
+	Items = "Itens",
+	Discounts = "Descontos",
+	Costs = "Custos",
+	CostTotal = "Custo Total",
+	CostBag = "Custo dos Itens na Bolsa",
+	CostEquip = "Custo dos Itens Equipados",
+	TooltipOptions = "Tooltip",
+};
+
+L["TITAN_REPAIR"] = "Titan Consertos"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "Guild Bank Funds :"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "Guild Bank Withdrawal Allowed :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "Use Guild Bank Funds"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "Guild Bank can't afford the repair cost, or you can't withdraw that much."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "You are either not in a guild or you don't have permission to use the guild bank to repair your items."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "You cannot afford to repair, at this time."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "Report Repair Cost to Chat"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "Custo de conserto foi "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "Total Gold on";
+L["TITAN_GOLD_ITEMNAME"] = "Titan Gold";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "Limpar Banco de Dados";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "Resetar Sessão Atual";
+L["TITAN_GOLD_DB_CLEARED"] = "Titan Gold - Database Cleared.";
+L["TITAN_GOLD_SESSION_RESET"] = "Titan Ouro - Sessão resetada.";
+L["TITAN_GOLD_MENU_TEXT"] = "Ouro";
+L["TITAN_GOLD_TOOLTIP"] = "Informações de Ouro";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "Display Player Gold";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "Display Server Gold";
+L["TITAN_GOLD_SESS_EARNED"] = "Ganho nesta Sessão";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "Ganho por Hora";
+L["TITAN_GOLD_SESS_LOST"] = "Perda Nesta Sessão";
+L["TITAN_GOLD_PERHOUR_LOST"] = "Perda por Hora";
+L["TITAN_GOLD_STATS_TITLE"] = "Estatísticas da Sessão";
+L["TITAN_GOLD_TTL_GOLD"] = "Ouro Total";
+L["TITAN_GOLD_START_GOLD"] = "Ouro Inicial";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "Ordernar Tabela por Ouro";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "Ordenar Ouro por Nome";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "Exibir Ouro por Hora";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "Ocultar Gold Per Hour";
+L["TITAN_GOLD_GOLD"] = "o";
+L["TITAN_GOLD_SILVER"] = "p";
+L["TITAN_GOLD_COPPER"] = "c";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "Visível";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "Escondido";
+L["TITAN_GOLD_DELETE_PLAYER"] = "Apagar Personagem";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Exibir Personagem";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "Aliança";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "Horda";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE .. "Atenção: "
+.. FONT_COLOR_CODE_CLOSE .. "This setting will wipe your Titan Gold database. "
+.. "If you wish to continue with this operation, push 'Accept', otherwise push 'Cancel' or the 'Escape' key.";
+L["TITAN_GOLD_COIN_NONE"] = "Exibir Nenhum Rótulo";
+L["TITAN_GOLD_COIN_LABELS"] = "Exibir Rótulo de Texto";
+L["TITAN_GOLD_COIN_ICONS"] = "Exibir Rótulo de Ícones";
+L["TITAN_GOLD_ONLY"] = "Exibir Somente Ouro";
+L["TITAN_GOLD_COLORS"] = "Exibir Gold Colors";
+L["TITAN_GOLD_MERGE"] = "Merge Servers";
+L["TITAN_GOLD_SEPARATE"] = "Separate Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "Informação de Volume";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "Master Sound Volume: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "Effects Sound Volume: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "Ambience Sound Volume: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "Music Sound Volume: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "Microphone Sound Volume: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "Speaker Sound Volume: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "Hint: Clique para ajustar o"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "volume do som.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "Volume Control: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "Volume Control";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "Geral";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "Efeitos";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "Ambiente";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "Música";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "Microfone";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "Auto-falante";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "Alto";
+L["TITAN_VOLUME_CONTROL_LOW"] = "Baixo";
+L["TITAN_VOLUME_MENU_TEXT"] = "Controle do Volume";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "Exibir Opções de Som/Voz";
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "Ignorar Configurações de Volume da Blizzard";
diff --git a/Titan/locale/Localization.CN.lua b/Titan/locale/Localization.CN.lua
new file mode 100644
index 0000000..a848b73
--- /dev/null
+++ b/Titan/locale/Localization.CN.lua
@@ -0,0 +1,556 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","zhCN")
+if not L then return end
+
+L["TITAN_PANEL"] = "Titan Panel";
+local TITAN_PANEL = "Titan Panel";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/A";
+L["TITAN_SECONDS"] = "秒";
+L["TITAN_MINUTES"] = "分";
+L["TITAN_HOURS"] = "小时";
+L["TITAN_DAYS"] = "天";
+L["TITAN_SECONDS_ABBR"] = "秒";
+L["TITAN_MINUTES_ABBR"] = "分";
+L["TITAN_HOURS_ABBR"] = "小时";
+L["TITAN_DAYS_ABBR"] = "天";
+L["TITAN_MILLISECOND"] = "毫秒";
+L["TITAN_KILOBYTES_PER_SECOND"] = "KB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kb/s"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "无";
+L["TITAN_USE_COMMA"] = "千位分隔符使用逗号";
+L["TITAN_USE_PERIOD"] = "使用点";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "无法删除你的当前配置.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."警告 : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "如果你想继续这个操作, 按 '接受' (会重载界面), 否则请按 '取消' ."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."这会重置面板的位置和"..TITAN_PANEL.." 设置信息为默认值,并重建配置文件. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."这会重新加载"..TITAN_PANEL..". "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = TITAN_PANEL.."注册请求" --??
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "其他插件注册请求" --??
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "下列插件扩展要注册在"..TITAN_PANEL.."下.\n"
+	.."如有相关问题请发送给其插件作者."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "类型" --??
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "类别"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "按钮名称"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "状态"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "信息"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "说明"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "表单"
+L["TITAN_PANEL_EXTRAS"] = TITAN_PANEL.." 附加"
+L["TITAN_PANEL_EXTRAS_SHORT"] = "附加"
+L["TITAN_PANEL_EXTRAS_DESC"] = "这些是有设置数据但没有加载的扩展组件.\n"
+	.."可以安全的删除."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "删除设置数据"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "设置数据已经被删除."
+L["TITAN_PANEL_CHARS"] = "角色"
+L["TITAN_PANEL_CHARS_DESC"] = "这些是有设置数据的角色."
+L["TITAN_PANEL_REGISTER_START"] = "正在注册 "..TITAN_PANEL.." 扩展组件..."
+L["TITAN_PANEL_REGISTER_END"] = "注册完成."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."使用命令: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffffResets "..TITAN_PANEL.."重置为默认设置.";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffResets "..TITAN_PANEL.." 重置提示文字字体缩放为默认值.";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffResets "..TITAN_PANEL.." 重置提示文字透明度为默认值.";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffResets "..TITAN_PANEL.." 重置面板缩放为默认值.";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffResets "..TITAN_PANEL.." 重置按钮间距为默认值.";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."使用命令: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffff打开 "..TITAN_PANEL.." 控制面板.";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffff打开透明度控制面板.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffff打开皮肤控制面板.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."使用命令: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <名字> <服务器>: |cffffffff使用此服务器下的此角色的配置设置.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<名字>: |cffffffff可以是角色名也可以是配置文件名."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<服务器>: |cffffffff可以是服务器名也可以是 Titan配置模板名."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffToggles "..TITAN_PANEL.." to load silently.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."使用命令: |cffffffff/titan {help | help <主题>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<主题>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."使用命令: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<主题>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.."Titan Panel 提示文字缩放已重置.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.."Titan Panel 提示窗口的透明度已重置.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.."Titan Panel 缩放已重置.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.."Titan Panel 按钮间距已重置.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "全局";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "全局陪置";
+L["TITAN_PANEL_GLOBAL_USE"] = "使用全局配置";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "设为全局配置";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "为所有角色使用全局配置";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "重置选项";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "当一个全局配置文件在使用中时你可以无法加载";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "版本|cffffd700 |cffff8c00"..TITAN_PANEL.." 开发团队出品";
+L["TITAN_PANEL_MENU_TITLE"] = TITAN_PANEL;
+L["TITAN_PANEL_MENU_HIDE"] = "隐藏";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(战斗中)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(将重载界面)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "显示彩色文本";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "显示图标";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "显示名称";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "自动隐藏";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "文字居中";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "显示Titan条";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "禁用自动适应屏幕";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "禁用微缩地图自动出现";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "自动记录";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "自动背包调整";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "自动调整标签框体";
+L["TITAN_PANEL_MENU_PROFILES"] = "配置";
+L["TITAN_PANEL_MENU_PROFILE"] = "配置";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "个人";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " 已删除.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "服务器";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "角色";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "按下 '确定' 界面将重载来保存你的个人配置.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "为你的配置文件输入一个名称:\n(20字符限制,不能有空格)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "现有设置已经被保存为配置文件: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "配置文件名称已存在. 你确定要覆盖它? 按 '接受' 确定, 按 '取消' 取消.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "加载配置";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "载入";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "删除";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "保存";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "配置设置";
+L["TITAN_PANEL_OPTIONS"] = "选项";
+L["TITAN_PANEL_MENU_TOP"] = "顶端面板条"
+L["TITAN_PANEL_MENU_TOP2"] = "顶端面板条2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "底端面板条"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "底端面板条2"
+L["TITAN_PANEL_MENU_OPTIONS"] = TITAN_PANEL.." 提示文字和框架";
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "提示文字和框架";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "顶端面板条"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "底端面板条"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "面板条"
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = TITAN_PANEL.." 顶端面板条";
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = TITAN_PANEL.." 底端面板条";
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "提示文字";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "框架";
+L["TITAN_PANEL_MENU_PLUGINS"] = "扩展组件";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "锁定按钮";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "显示扩展组件版本";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "右侧插件"; --??
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "强制LDB启动器到右侧";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"扩展组件","综合","战斗","信息","用户界面","专业技能"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "显示提示文字";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "在战斗中隐藏提示文字";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "在战斗中锁定自动隐藏的面板条";
+L["TITAN_PANEL_MENU_RESET"] = "重置 "..TITAN_PANEL.." 为默认";
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "皮肤";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "面板字体"
+L["TITAN_PANEL_MENU_ENABLED"] = "启用";
+L["TITAN_PANEL_MENU_DISABLED"] = "禁用";
+L["TITAN_PANEL_SHIFT_LEFT"] = "向左移动";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "向右移动";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "显示扩展组件文本";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "总是显示"; --??
+L["TITAN_PANEL_MENU_POSITION"] = "位置";
+L["TITAN_PANEL_MENU_BAR"] = "面板条";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "选择扩展组件显示在哪个面板条上";
+L["TITAN_PANEL_MENU_SHOW"] = "显示扩展组件";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "刷新扩展组件";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "刷新扩展组件的文本和位置";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Silent Load";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "版本";
+L["TITAN_ABOUT_AUTHOR"] = "作者";
+L["TITAN_ABOUT_CREDITS"] = "贡献者";
+L["TITAN_ABOUT_CATEGORY"] = "类别";
+L["TITAN_ABOUT_EMAIL"] = "Email";
+L["TITAN_ABOUT_WEB"] = "网址";
+L["TITAN_ABOUT_LICENSE"] = "许可证";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "信息条插件. 在屏幕顶部或底部增加一个信息面板,允许用户添加显示所需信息或者快速载入其他插件.";
+L["TITAN_TRANS_MENU_TEXT"] = TITAN_PANEL.." 透明度";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "透明度";
+L["TITAN_TRANS_MENU_DESC"] = "为 "..TITAN_PANEL.." 面板条和提示文字调整透明度.";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "主面板条";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "辅面板条";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "提示文字";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "为扩展组件和其提示文字调整透明度.";
+L["TITAN_UISCALE_MENU_TEXT"] = TITAN_PANEL.." 缩放和字体";
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "缩放和字体";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "用户界面缩放";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = TITAN_PANEL.." 缩放";
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "按钮间距";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "图标间距";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "提示文字缩放";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "禁用提示文字缩放";
+L["TITAN_UISCALE_MENU_DESC"] = "控制 "..TITAN_PANEL.."的界面.";
+L["TITAN_UISCALE_SLIDER_DESC"] = "设置你整体用户界面的缩放.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "控制 "..TITAN_PANEL.." 的各个按钮图标的缩放.";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "调整左侧的扩展组件间距.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "调整右侧的扩展组件间距.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "调整各个扩展组件提示文字的缩放.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "禁用"..TITAN_PANEL.." 提示文字缩放.";
+
+L["TITAN_SKINS_TITLE"] = TITAN_PANEL.." 皮肤";
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "皮肤 - 自定义";
+L["TITAN_SKINS_TITLE_CUSTOM"] = TITAN_PANEL.." 自定义皮肤";
+L["TITAN_SKINS_MAIN_DESC"] = "所有自定义皮肤都放在以下文件夹中: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ ".."\n"
+			.."\n"..TITAN_PANEL.."自定义皮肤寸放在各自账号的插件配置文件夹.";
+L["TITAN_SKINS_LIST_TITLE"] = "皮肤列表";
+L["TITAN_SKINS_SET_DESC"] = "给"..TITAN_PANEL.."信息条选择一个皮肤.";
+L["TITAN_SKINS_SET_HEADER"] = "设置面板皮肤";
+L["TITAN_SKINS_RESET_HEADER"] = "重置"..TITAN_PANEL.." 皮肤";
+L["TITAN_SKINS_NEW_HEADER"] = "添加新皮肤";
+L["TITAN_SKINS_NAME_TITLE"] = "皮肤名称";
+L["TITAN_SKINS_NAME_DESC"] = "为新皮肤输入一个名称.";
+L["TITAN_SKINS_PATH_TITLE"] = "皮肤路径";
+L["TITAN_SKINS_PATH_DESC"] = "输入皮肤的准确路径, 如范例所示.";
+L["TITAN_SKINS_ADD_HEADER"] = "添加皮肤";
+L["TITAN_SKINS_ADD_DESC"] = "添加一个新皮肤到面板可用皮肤列表.";
+L["TITAN_SKINS_REMOVE_HEADER"] = "删除皮肤";
+L["TITAN_SKINS_REMOVE_DESC"] = "从面板可用皮肤列表删除一个皮肤.";
+L["TITAN_SKINS_REMOVE_BUTTON"] = "删除";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "从面板可用皮肤列表删除一个皮肤.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "你需要从"
+..TITAN_PANEL.." 安装目录操作,来彻底删除你不需要的皮肤. 插件无法添加或删除文件."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "恢复默认";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "恢复皮肤列表至默认值.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "选择Titan条上各个模块的字体样式.";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "字体大小";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "设置面板上的字体大小.";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = "面板框架层叠";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "设置"..TITAN_PANEL.."的框架是否显示在最前.";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "高级";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "当你遇到页面框架出错时更改计时器.".."\n"; --??
+L["TITAN_PANEL_MENU_ADV_PEW"] = "登陆界面"; --??
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "当你登陆登出游戏或者进出副本时遇到页面框架出错时更改设定值(通常是增大).";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "坐骑";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "当你上下坐骑遇到页面框架出错时更改设定值(通常是增大).";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "控制"..TITAN_PANEL.."自动隐藏 开/关";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "背包: ";
+L["TITAN_BAG_TOOLTIP"] = "背包状态";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "提示: 左键点击打开所有背包.";
+L["TITAN_BAG_MENU_TEXT"] = "背包监视";
+L["TITAN_BAG_USED_SLOTS"] = "已用空间";
+L["TITAN_BAG_FREE_SLOTS"] = "剩余空间";
+L["TITAN_BAG_BACKPACK"] = "背包";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "显示已用空间";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "显示可用空间";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "显示详细的提示信息";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "忽略容器";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "忽略专业技能包";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"魔化魔纹布包", "魔化符文布包", "附魔师之袋", "大附魔袋", "魔焰背包",
+"神秘背包", "异界之袋", "“马车——限定款”附魔之夜手提包"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"重工具箱", "魔铁工具箱", "泰坦神铁工具箱", "氪金工具箱", "源质工具箱", "“马车——狂人”高科技背包"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"草药袋", "塞纳里奥草药包", "塞纳留斯之袋", "麦卡的草药包", "翡翠包", "海加尔远征背囊",
+"“马车——拥抱绿野”草药手提袋"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"皇家铭文师背包", "无尽口袋", "“马车——珊德拉”学徒手提包", "Burnished Inscription Bag"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"宝石袋", "珠宝袋", "“马车——限定款”宝石镶嵌挎包"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"制皮匠的背包", "大皮袋", "猎户的旅行背包", "“马车——米亚”真皮背包", "Burnished Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"矿物包", "加固矿工袋", "猛犸皮矿石包", "三重加固的矿工袋", "“马车——克莉斯汀娜”珍藏金属背包", "Burnished Mining Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"捕鱼大师的工具箱"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"便携式冷柜", "Advanced Refrigeration Unit"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "时钟";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "与服务器的时差: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "本地时间: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "服务器时间: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "修正服务器时间: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "提示: 左键单击来修正时间"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "(仅服务器时间) 24小时模式.";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Shift+左键单击 打开/关闭日历.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "服务器时差: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "时差";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24小时制";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "切换 12/24 小时制显示";
+L["TITAN_CLOCK_MENU_TEXT"] = "时钟";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "显示本地时间";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "显示服务器时间";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "显示修正后的服务器时间";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "在最右侧显示";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "隐藏 时间/日历 按钮";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "隐藏时间按钮";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "隐藏日历按钮";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "坐标格式";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "位置: ";
+L["TITAN_LOCATION_TOOLTIP"] = "所在位置信息";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "提示: Shift + 左键单击添加所在位置";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "信息到聊天窗口.";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "区域: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "具体地点: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "PVP 信息: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "炉石位置";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "旅店: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "坐标";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "显示区域信息";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "在世界地图上显示坐标";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "鼠标位置(X,Y): %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "玩家位置(X,Y): %s";
+L["TITAN_LOCATION_NO_COORDS"] = "无坐标";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "在小地图显示位置";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "当切换地区时更新世界地图";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "FPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "当前FPS: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "平均 FPS: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "最低 FPS: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "最高 FPS: ";
+L["TITAN_FPS_TOOLTIP"] = "每秒画面帧数";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "延迟: ";
+L["TITAN_LATENCY_TOOLTIP"] = "网络状况信息";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "游戏延迟 (本地): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "游戏延迟 (世界): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "接收带宽: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "发送带宽: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "延迟";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "分配: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "自由拾取";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "轮流拾取";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "队长分配";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "队伍分配";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "需求优先";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "分配方式";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "分配方式";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "Roll点";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "左键单击将Roll点.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "右键点击选择Roll点类型.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "五人副本难度";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "团队副本难度";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "显示副本难度";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "设置五人副本难度";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "设置团队副本难度";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "自动设定(基于队伍类型)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "内存: ";
+L["TITAN_MEMORY_TOOLTIP"] = "内存使用";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "当前: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "起始: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "增长率: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "性能信息";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "性能";
+L["TITAN_PERFORMANCE_ADDONS"] = "插件使用";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "插件的内存使用";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "插件内存占用形式";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "插件CPU使用";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "名称:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "使用";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "百分比";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "插件使用内存总量:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "CPU使用总量:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "显示 FPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "显示本地延迟";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "显示世界延迟";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "显示内存使用";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "显示插件内存使用量";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "显示插件内存使用率";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "CPU 使用图形显示模式";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "启用CPU图形显示 ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "禁用CPU图形显示 ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "监视的插件: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "被监视的插件";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "左键点击回收内存.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "经验/小时 当前等级: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "经验/小时 本次连接: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "升级所需时间: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "升级: ";
+L["TITAN_XP_TOTAL_RESTED"] = "剩余: ";
+L["TITAN_XP_XPTOLEVELUP"] = "升级所需经验: ";
+L["TITAN_XP_TOOLTIP"] = "经验相关信息";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "总共游戏时间: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "当前等级的游戏时间: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "本次连接游戏时间: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "当前级别总经验: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "当前等级所获经验: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "升级所需经验: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "本次连接所获经验: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "经验/小时 当前级别: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "经验/小时 本次连接: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "升级时间 (按等级效率): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "升级时间 (按连接效率): ";
+L["TITAN_XP_MENU_TEXT"] = "经验";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "显示当前级别 经验/小时";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "显示本次连接 经验/小时";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "更多信息显示";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "按钮";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "显示双倍的经验值";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "显示升级所需经验";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "显示升级所需的预估击杀数";
+L["TITAN_XP_MENU_RESET_SESSION"] = "重置连接时间";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "刷新计时器";
+L["TITAN_XP_UPDATE_PENDING"] = "更新中...";
+L["TITAN_XP_KILLS_LABEL"] = "升级所需击杀数 (基于最后一个的经验 %s): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "预估击杀数: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "连接时间: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "显示连接时间";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*)死亡,你获得(%d+)点经验。";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "预估获得经验: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "升级所需杀怪数 (基于最后杀怪所获%s点经验): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "显示升级所需(基于最后一次所获经验)";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "修理",
+	tooltip = "耐久度信息",
+	button = "耐久度: ",
+	normal = "修理费用 (正常): ",
+	friendly = "修理费用 (友善): ",
+	honored = "修理费用 (尊敬): ",
+	revered = "修理费用 (崇敬): ",
+	exalted = "修理费用 (崇拜): ",
+	buttonNormal = "显示正常费用",
+	buttonFriendly = "显示(声望友善)费用 (5%折扣)",
+	buttonHonored = "显示(声望尊敬)费用 (10%折扣)",
+	buttonRevered = "显示(声望崇敬)费用 (15%折扣)",
+	buttonExalted = "显示(声望崇拜)费用 (20%折扣)",
+	percentage = "显示为百分比",
+	itemnames = "显示物品名称",
+	mostdamaged = "耐久度最低的",
+	showdurabilityframe = "显示耐久度面板",
+	undamaged = "显示未掉耐久度的物品",
+	discount = "折扣",
+	nothing = "没有需要修理的物品",
+	confirmation = "你确定要修理所有装备吗 ?",
+	badmerchant = "这个商人不能修理,现在将显示正常费用",
+	popup = "显示维修框",
+	showinventory = "统计背包中的需修理物品",
+	WholeScanInProgress = "更新中...",
+	AutoReplabel = "自动修理",
+	AutoRepitemlabel = "自动修理所有物品",
+	ShowRepairCost = "显示修理费用",
+	ignoreThrown = "忽略投掷武器",
+	ShowItems = "显示物品",
+	ShowDiscounts = "显示折扣",
+	ShowCosts = "显示花费",
+	Items = "物品",
+	Discounts = "折扣",
+	Costs = "花费",
+	CostTotal = "总花费",
+	CostBag = "背包内花费",
+	CostEquip = "身上装备花费",
+	TooltipOptions = "提示文字",
+};
+
+L["TITAN_REPAIR"] = "Titan 修理"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "公会银行总资金 :"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "公会银行允许的费用 :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "使用公会银行资金"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "公会银行余额不足或者你超过了允许的使用量."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "你没有公会或你没有权限使用公会资金来修理."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "现在,至少,你没钱来修理."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "报告修理费到聊天频道"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "修理花费 "
+
+L["TITAN_PLUGINS_MENU_TITLE"] = "扩展模块";
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "统计所持有金币信息";
+L["TITAN_GOLD_ITEMNAME"] = "金币助手";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "清除已存数据";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "重置现有周期";
+L["TITAN_GOLD_DB_CLEARED"] = "Titan金币查看助手 - 数据已清除.";
+L["TITAN_GOLD_SESSION_RESET"] = "Titan金币查看助手 - 周期已重置.";
+L["TITAN_GOLD_MENU_TEXT"] = "金币助手";
+L["TITAN_GOLD_TOOLTIP"] = "金币信息";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "显示玩家金币";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "显示所有角色金币";
+L["TITAN_GOLD_SESS_EARNED"] = "这个周期所赚金币";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "每小时所赚金币";
+L["TITAN_GOLD_SESS_LOST"] = "这个周期所花费金币";
+L["TITAN_GOLD_PERHOUR_LOST"] = "每小时花费金币";
+L["TITAN_GOLD_STATS_TITLE"] = "周期统计";
+L["TITAN_GOLD_TTL_GOLD"] = "总金币";
+L["TITAN_GOLD_START_GOLD"] = "起始金币数";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "按金币数排列";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "按角色名排列";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "显示每小时金币进出";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "隐藏每小时金币进出";
+L["TITAN_GOLD_GOLD"] = "g";
+L["TITAN_GOLD_SILVER"] = "s";
+L["TITAN_GOLD_COPPER"] = "c";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "显示";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "隐藏";
+L["TITAN_GOLD_DELETE_PLAYER"] = "删除这个角色";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Show toon";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "联盟";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "部落";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."警告: "
+..FONT_COLOR_CODE_CLOSE.."这会清楚你的Titan金币统计数据库. "
+.."如果你想继续操作请按 '接受', 否则请按 '取消'.";
+L["TITAN_GOLD_COIN_NONE"] = "不现实标签";
+L["TITAN_GOLD_COIN_LABELS"] = "显示标签文字";
+L["TITAN_GOLD_COIN_ICONS"] = "显示标签图标";
+L["TITAN_GOLD_ONLY"] = "只显示金";
+L["TITAN_GOLD_COLORS"] = "显示金币颜色";
+L["TITAN_GOLD_MERGE"] = "Merge Servers";
+L["TITAN_GOLD_SEPARATE"] = "Separate Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "音量信息";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "主音量: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "音效音量: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "环境音量: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "音乐音量: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "麦克风音量: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "扬声器音量: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "提示: 用鼠标左键来调节"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "音量.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "音量控制: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "音量控制";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "主音量";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "音效音量";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "环境音量";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "音乐";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "麦克风";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "扬声器";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "高";
+L["TITAN_VOLUME_CONTROL_LOW"] = "低";
+L["TITAN_VOLUME_MENU_TEXT"] = "音量控制";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "显示 声音控制" ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "替换默认声音控制";
diff --git a/Titan/locale/Localization.DE.lua b/Titan/locale/Localization.DE.lua
new file mode 100644
index 0000000..4e870f5
--- /dev/null
+++ b/Titan/locale/Localization.DE.lua
@@ -0,0 +1,552 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","deDE")
+if not L then return end
+
+L["TITAN_PANEL"] = "Titan Panel";
+local TITAN_PANEL = "Titan Panel";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/V";
+L["TITAN_SECONDS"] = "Sekunde(n)";
+L["TITAN_MINUTES"] = "Minute(n)";
+L["TITAN_HOURS"] = "Stunde(n)";
+L["TITAN_DAYS"] = "Tag(e)";
+L["TITAN_SECONDS_ABBR"] = "s";
+L["TITAN_MINUTES_ABBR"] = "m";
+L["TITAN_HOURS_ABBR"] = "S";
+L["TITAN_DAYS_ABBR"] = "T";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "KB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kbps"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "None";
+L["TITAN_USE_COMMA"] = "Benutze Komma";
+L["TITAN_USE_PERIOD"] = "Benutze Punkt";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "Aktuelles Profil darf nicht gel\195\182scht werden.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."WARNUNG : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "Wenn Du diesen Schritt ausführen willst, klicke auf 'Accept' (deine UI wird neu geladen), anderenfalls klicke auf 'Abbrechen' oder dr\195\188cke die 'ESC'-Taste."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."Diese Einstellung wird die Titanbars und "..TITAN_PANEL.." Einstellungen auf die Standardwerte zur\195\188cksetzen. Zus195\164tzlich wird das Profil neu erstellt."
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."Dies wird "..TITAN_PANEL.." neu laden. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = TITAN_PANEL.." Versuche"
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Versuche"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "Folgende Plugins haben eine Registrierungsanfrage an "..TITAN_PANEL.." gestellt.\n"
+	.."Bitte sende entsprechende Fehler an deren Autor."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Art"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Kategorie"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Schaltfl195\164che"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Status"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Grund"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Hinweise"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Tabellenindex"
+L["TITAN_PANEL_EXTRAS"] = TITAN_PANEL.." Extras"
+L["TITAN_PANEL_EXTRAS_SHORT"] = "Extras"
+L["TITAN_PANEL_EXTRAS_DESC"] = "Dies sind Plugins, deren Konfigurationsdaten nicht geladen wurden.\n"
+	.."Diese k\195\182nnen gefahrlos gel\195\182scht werden."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "L\195\182sche Konfigurationsdaten"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "Konfigurationseintrag wurde gel\195\182scht."
+L["TITAN_PANEL_CHARS"] = "Spielercharaktere"
+L["TITAN_PANEL_CHARS_DESC"] = "Dies sind Spielercharaktere mit Konfigurationsdaten."
+L["TITAN_PANEL_REGISTER_START"] = "Registriere "..TITAN_PANEL.." Plugins..."
+L["TITAN_PANEL_REGISTER_END"] = "Registrationsprozess abgeschlossen."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Benutzen: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffffSetzt Panelposition/-inhalt auf Voreinstellung zur\195\188ck.";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffSetzt Paneltooltip-Schriftgr\195\182\195\159e auf Voreinstellung zur\195\188ck.";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffSetzt Paneltooltip-Transparenz auf Voreinstellung zur\195\188ck.";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffSetzt Panelgr\195\182\195\159e auf Voreinstellung zur\195\188ck.";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffSetzt Panel-Buttonabstand auf Voreinstellung zur\195\188ck.";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Benutzen: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffff\195\150ffnet das Ace3-Panel-Kontroll-GUI.";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffff\195\150ffnet das Ace3-Transparenz-Kontroll-GUI.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffff\195\150ffnet das Ace3-Skin-Kontroll-GUI.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Benutzen: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <name> <server>: |cffffffff\195\156berschreibt das aktuelle Profil mit dem angegebenen, gespeicherten Profil.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<name>: |cffffffffkann entweder ein Spiekercharatername oder ein individuell bezeichnetes Profil sein."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<server>: |cffffffffkann entweder ser Servername oder 'TitanCustomProfile' sein."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Benutzen: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffSchaltet "..TITAN_PANEL.." so um, dass es still geladen wird.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Benutzen: |cffffffff/titan {Ordenshalle}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."Ordenshalle: |cffffffffSchaltet die "..TITAN_PANEL.." Ordenshalle Befehlsleiste um.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Benutzen: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<Punkt>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Benutzen: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<Punkt>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel Tooltip-Schriftgr\195\182sse wurde zur\195\188ckgesetzt.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel Tooltip-Transparenz wurde zur\195\188ckgesetzt.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel Gr\195\182sse wurde zur\195\188ckgesetzt.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel Buttonabstand wurde zur\195\188ckgesetzt.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Global";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Global-Profil";
+L["TITAN_PANEL_GLOBAL_USE"] = "Benutze Global-Profil";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Benutze als Global-Profil";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Benutze ein Global-Profil f\195\188r alle Spielercharaktere";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "Setze Einstellungen zur\195\188ck";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "Du kannst kein individuelles Profil laden, wenn ein globales Profil verwendet wird.";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffffd700 vom |cffff8c00"..TITAN_PANEL.." Development Team";
+L["TITAN_PANEL_MENU_TITLE"] = TITAN_PANEL;
+L["TITAN_PANEL_MENU_HIDE"] = "Verstecken";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(Im Kampf)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(UI neuladen)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "Farbigen Text anzeigen";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "Icon anzeigen";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "Beschriftungstext anzeigen";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "Titanleiste automatisch ausblenden";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "Text zentrieren";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Zeige Leiste";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "Bildschirmjustierung deaktivieren (Gr\195\182\195\159en\195\164nderung)";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "Minimapjustierung deaktivieren";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "Log automatisch justieren";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "Automatische Taschenanpassung";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "Automatische Ticketfensteranpassung";
+L["TITAN_PANEL_MENU_PROFILES"] = "Profile";
+L["TITAN_PANEL_MENU_PROFILE"] = "Profil ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "Benutzerdefiniert";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " wurde gel\195\182scht.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "Server";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "Charakter";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "UI wird jetzt neu geladen nach dr\195\188cken von 'Okay' und das benutzerdefinierte Profil sichern.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "Name f\195\188r benutzerdefiniertes Profil:\n(20 Zeichen max, keine Leerzeichen erlaubt, Gross-/Kleinschreibung beachten) ";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "Aktuelle Leisteneinstellung werden gesichtert unter Profilname: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "Dieser Profilname existiert bereits. Soll wirklich \195\188berschrieben werden? Dr\195\188cke 'Accept' f\195\188r JA, wenn NEIN dr\195\188cke 'Cancel' oder 'Escape'.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "Verwalten";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "Einstellungen laden";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "L\195\182schen";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "Sichern";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "Konfiguration";
+L["TITAN_PANEL_OPTIONS"] = "Optionen";
+L["TITAN_PANEL_MENU_TOP"] = "Oben"
+L["TITAN_PANEL_MENU_TOP2"] = "Oben 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "Unten"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "Unten 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = TITAN_PANEL.." Tooltips und Rahmen";
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Tooltips und Rahmen";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Obere Leisten"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Untere Leisten"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "Leisten";
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = TITAN_PANEL.." - Obere Leisten";
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = TITAN_PANEL.." - Untere Leisten";
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Tooltips";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "Bilder pro Sek.";
+L["TITAN_PANEL_MENU_PLUGINS"] = "Plugins";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "Schaltfl\195\164chen fixieren";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "Pluginversionen anzeigen";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "Plugin rechts";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "Schiebe Starter nach rechts";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"Titan Standardplugins","Allgemein","Kampf","Informationen","Interface","Beruf"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "Zeige Tooltips";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "Verberge Tooltips im Kampf";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Sperre Auto-Verbergen der Leisten, wenn im Kampf";
+L["TITAN_PANEL_MENU_RESET"] = "Einstellungen zur\195\188cksetzen";
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "Skins";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "Panel-Schriftart"
+L["TITAN_PANEL_MENU_ENABLED"] = "Ein";
+L["TITAN_PANEL_MENU_DISABLED"] = "Aus";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Shift-Links";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Shift-Rechts";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Zeige Plugintext";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Immer An";
+L["TITAN_PANEL_MENU_POSITION"] = "Position";
+L["TITAN_PANEL_MENU_BAR"] = "Leiste";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Festlegen, in welcher Leiste das Plugin angezeigt wird.";
+L["TITAN_PANEL_MENU_SHOW"] = "Zeige Plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Aktualisiere Plugins";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Aktualisiere Plugin-Text und Position";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Stilles Laden";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Ordenshalle Befehlsleiste verstecken";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Version";
+L["TITAN_ABOUT_AUTHOR"] = "Autor";
+L["TITAN_ABOUT_CREDITS"] = "Credits";
+L["TITAN_ABOUT_CATEGORY"] = "Kategorie";
+L["TITAN_ABOUT_EMAIL"] = "Email";
+L["TITAN_ABOUT_WEB"] = "Website";
+L["TITAN_ABOUT_LICENSE"] = "Lizenz";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "Informationsleisten-Addon. Erlaubt Dateneingaben oder Start-Zus\195\164tze auf einer Kontrollleiste am oberen oder unteren Bildschirmrand.";
+L["TITAN_TRANS_MENU_TEXT"] = TITAN_PANEL.." Transparenz";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "Transparenz";
+L["TITAN_TRANS_MENU_DESC"] = "Transparenz f\195\188r Titanleisten und Tooltips einstellen.";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "Hauptleiste";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "Erweiterungsleiste";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "Tooltip";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "Setzt Transparenz der Tooltips verschiedener Plugins.";
+L["TITAN_UISCALE_MENU_TEXT"] = "Leisteneinstellungen";
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Scale and Font";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "UI Gr\195\182sse";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "Leistengr\195\182sse";
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "Buttonabstand";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Iconabstand";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "Tooltip Font-Skalierung" ;
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "Tooltip Font-Skalierung aus";
+L["TITAN_UISCALE_MENU_DESC"] = "Kontrolliert verschiedene Aspekte von der UI und den Leisten.";
+L["TITAN_UISCALE_SLIDER_DESC"] = "Skaliert das gesamte UI.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "Skaliert verschiedene Panelbuttons und Icons.";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "Abstandsjustierung linksausgerichtete Plugins.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Abstandsjustierung rechtsausgerichtete Plugins.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "Gr\195\182ssenjustierung f\195\188r Tooltips verschiedener Plugins.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "Titan's Tooltip Font-Skalierung AUS.";
+
+L["TITAN_SKINS_TITLE"] = TITAN_PANEL.." Skins";
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Skins - Eigene";
+L["TITAN_SKINS_TITLE_CUSTOM"] = TITAN_PANEL.." Eigene Skins";
+L["TITAN_SKINS_MAIN_DESC"] = "Alle eigenen Skins sollten in: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ sein".."\n"
+			.."\n"..TITAN_PANEL.." und angepasste skins sind im Custom Ordner abgelegt."
+L["TITAN_SKINS_LIST_TITLE"] = "Skin-Auswahl";
+L["TITAN_SKINS_SET_DESC"] = "W\195\164hle Skin f\195\188r die Titanleisten.";
+L["TITAN_SKINS_SET_HEADER"] = "Bestimme Leisten-Skin";
+L["TITAN_SKINS_RESET_HEADER"] = "Reset "..TITAN_PANEL.." Skins";
+L["TITAN_SKINS_NEW_HEADER"] = "Ne\195\188 Skin hinzuf\195\188gen";
+L["TITAN_SKINS_NAME_TITLE"] = "Skin Name";
+L["TITAN_SKINS_NAME_DESC"] = "Name f\195\188r neue Skin.";
+L["TITAN_SKINS_PATH_TITLE"] = "Skin Speicherort";
+L["TITAN_SKINS_PATH_DESC"] = "Exakten Speicherort der eigenen Skin eingeben, wie im Beispiel und in den Hinweisen beschrieben.";
+L["TITAN_SKINS_ADD_HEADER"] = "Skin hinzuf\195\188gen";
+L["TITAN_SKINS_ADD_DESC"] = "F\195\188gt eine neue Skin zur bestehenden Liste hinzu.";
+L["TITAN_SKINS_REMOVE_HEADER"] = "Entferne Skin";
+L["TITAN_SKINS_REMOVE_DESC"] = "Zu entfernende Skin aus der bestehenden Liste w\195\164hlen.";
+L["TITAN_SKINS_REMOVE_BUTTON"] = "Entfernen";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "Entfernt die gew\195\164hlte Skin aus der bestehenden Liste.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "Du bist eigenverantwortlich für das Entfernen von unerw\195\188nschten Skins "
+	.."aus dem "..TITAN_PANEL.." Installationsordner. Addons d\195\188rfen keine Dateien anlegen oder l\195\182schen."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "Zur\195\188cksetzen auf Standard";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "Setzt die Skinliste auf die Standard "..TITAN_PANEL.." Skins zur\195\188ck.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "Wähle die Schriftart f\195\188r die verschiedenen Plugins in den "..TITAN_PANEL.."-Leisten.";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "Font-Gr\195\182\195\159e";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "Legt die Font-Gr\195\182\195\159e f\195\188r "..TITAN_PANEL.." fest.";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = ""..TITAN_PANEL.." Frame-Strata";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "Legt die Frame-Strata f\195\188r die "..TITAN_PANEL.." Leiste(n) fest.";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Erweitert";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "\195\182ndere die Timereinstellungen nur, wenn es mit den Frameanordnungen Probleme gibt.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Betrete Spielwelt";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "\195\182ndere die Werte nur (normalerweise erh\195\182hen), wenn die Rahmen nach Betreten oder Verlassen der Spielwelt oder Instanzen nicht korrekt angezeigt werden.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Fahrzeug";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "\195\182ndere die Werte nur (normalerweise erh\195\182hen), wenn die Rahmen nach Betreten oder Verlassen eines Fahrzeugs nicht korrekt angezeigt werden.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "Leiste automatisch ausblenden";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "Taschen: ";
+L["TITAN_BAG_TOOLTIP"] = "Taschenbenutzung";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "Hinweis: Links-Klick um alle Taschen zu \195\182ffnen.";
+L["TITAN_BAG_MENU_TEXT"] = "Taschenbenutzung";
+L["TITAN_BAG_USED_SLOTS"] = "Benutzte Pl\195\164tze";
+L["TITAN_BAG_FREE_SLOTS"] = "Freie Pl\195\164tze";
+L["TITAN_BAG_BACKPACK"] = "Rucksack";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "Bereits belegte Pl\195\164tze anzeigen";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "Noch verf\195\188gbare Pl\195\164tze anzeigen";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "Zeige detaillierten Tooltip";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "Ignoriere Slots";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "Ignoriere Berufs-Taschen";
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Enchanted Mageweave Pouch", "Enchanted Runecloth Bag", "Enchanter's Satchel", "Big Bag of Enchantment", "Spellfire Bag",
+"Mysterious Bag", "Otherworldly Bag", "\"Carriage - Exclusive\" Enchanting Evening Purse"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Heavy Toolbox", "Fel Iron Toolbox", "Titanium Toolbox", "Khorium Toolbox", "Elementium Toolbox", "\"Carriage - Maddy\" High Tech Bag"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Herb Pouch", "Cenarion Herb Bag", "Satchel of Cenarius", "Mycah's Botanical Bag", "Emerald Bag", "Hyjal Expedition Bag",
+"\"Carriage - Going Green\" Herb Tote Bag"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Scribe's Satchel", "Pack of Endless Pockets", "\"Carriage - Xandera\" Student's Satchel"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Gem Pouch", "Bag of Jewels", "\"Carriage - Exclusive\" Gem Studded Clutch"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Leatherworker's Satchel", "Bag of Many Hides", "Trapper's Traveling Pack", "\"Carriage - Meeya\" Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Mining Sack", "Reinforced Mining Bag", "Mammoth Mining Bag", "\"Carriage - Christina\" Precious Metal Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Lure Master Tackle Box"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Portable Refrigerator"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "Uhr";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "Momentane Zeitverschiebung: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "Lokale Zeit: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "Server-Zeit: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "Eingestellte Server-Zeit: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "Hinweis: Links-Klick um die Zeitverschiebung festzulegen"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "und zwischen dem 12/24 Stundenformat zu wechseln.";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Shift Linksklick, um den Kalender ein- oder auszublenden.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "Zeitverschiebung in Stunden: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "Verschiebung";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24Std Fmt";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "Wechselt die Anzeige zwischen dem 12-Stunden und 24-Stunden Format.";
+L["TITAN_CLOCK_MENU_TEXT"] = "Uhr";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "Lokale Zeit anzeigen (L)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "Serverzeit anzeigen (S)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "Eingestellte Serverzeit anzeigen (A)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "Ganz rechts anzeigen.";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "Verstecke Zeit/Kalender-Knopf";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Verstecke Zeit-Knopf";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Verstecke Kalender-Knopf";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "Koordinatenformat";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "Pos: ";
+L["TITAN_LOCATION_TOOLTIP"] = "Info zur Position";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "Hinweis: Shift + Linksklick um die";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "momentane Position in den Chat einzuf\195\188gen.";-- Sensity
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "Zone: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "Sub-Zone: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "PVP-Info: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "Heimatort: ";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "Gasthaus: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "Position";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "Zonentext in der Leiste anzeigen";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "Koordinaten auf der Weltkarte anzeigen";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "Mauszeiger (X,Y): %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "Spieler (X,Y): %s";
+L["TITAN_LOCATION_NO_COORDS"] = "Keine Koordinaten";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "Zeige Gebutsname über der Minimap";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Aktualisiere Weltkarte bei Zonenwechsel";
+
+L["TITAN_FPS_FORMAT"] = "%.2f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "Bilder pro Sekunde (FPS)";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "Momentan: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "Durchschnitt: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "Bisher Minimum: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "Bisher Maximum: ";
+L["TITAN_FPS_TOOLTIP"] = "Bilder pro Sekunde";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.1f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "Latenz: ";
+L["TITAN_LATENCY_TOOLTIP"] = "Netzwerkstatus";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Realm-Latenz (Eigene): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Spiel-Latenz (Spielwelt): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "Bandbreite Eingang: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "Bandbreite Ausgang: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Latenz";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "Pl\195\188ndern: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "Jeder gegen Jeden (JGJ)";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "Reihum";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "Pl\195\188ndermeister";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "Pl\195\188ndern als Gruppe";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "Bedarf vor Gier";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "Pl\195\188ndermethode";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "Pl\195\188ndereinstellungen";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "W\195\188rfelwurf";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "Hinweis: Links-Klick f\195\188r W\195\188rfelwurf.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "Auswahl der W\195\188rfelmethode im Rrechtsklick-Men\195\188.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "Dungeon-Schwierigkeit";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "Schlachtzug-Schwierigkeit";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "Zeige Dungeon-/Schlachtzug-Schwierigkeit";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "Lege Dungeon-Schwierigkeit fest";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "Lege Schlachtzugschwierigkeit fest";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "Auto (gruppenbasiert)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.1f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.1f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "Speicher: ";
+L["TITAN_MEMORY_TOOLTIP"] = "Script Speichernutzung";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "Momentan: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "Anf\195\164nglich: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "Steigerungsrate: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "Leistung";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "Leistung";
+L["TITAN_PERFORMANCE_ADDONS"] = "Addon Nutzung";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "Addon Speicher Verbrauch";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "Addon Speicher Format";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "Addon CPU-Nutzung";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "Name:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "Verbrauch";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "Rate";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "Gesamt Addon Speichernutzung:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "Gesamt CPU-Leistung:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "Zeige FPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "Zeige Latenz";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Zeige Spielwelt-Latenz";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "Zeige Speicher";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "Zeige Addon Speicher Verbrauch";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "Zeige Addon Verbrauchs-Rate";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "CPU Profiling-Modus";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "CPU Profiling-Modus einschalten ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "CPU Profiling-Modus abschalten ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "\195\156berwachte Addons: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "\195\156berwachte Addons";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "Hinweis: Links-Klick f\195\188r eine Speicherbereinigung.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "XP/Std (Level): ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "XP/Std (Sitzung): ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "Zeit bis Aufstieg: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "Level komplett: ";
+L["TITAN_XP_TOTAL_RESTED"] = "Ausgeruht: ";
+L["TITAN_XP_XPTOLEVELUP"] = "XP bis Levelaufstieg: ";
+L["TITAN_XP_TOOLTIP"] = "Info XP";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "Spielzeit Gesamt: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "Spielzeit dieses Levels: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "Spielzeit dieser Sitzung: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "Gesamt XP dieses Levels: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "XP erhalten dieses Levels: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "XP ben\195\182tigt bis Aufstieg: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "XP erhalten diese Sitzung: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "XP/Std dieses Levels: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "XP/Std dieser Sitzung: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "Zeit bis Aufstieg (Levelrate): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "Zeit bis Aufstieg (Sitzungsrate): ";
+L["TITAN_XP_MENU_TEXT"] = "XP";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "Zeige XP/Std dieses Levels";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "Zeige XP/Std dieser Sitzung";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Zeige Multi-Info-Ansicht";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "Schaltfl\195\164che";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "Zeige XP (ausgeruht)";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "Zeige XP bis Levelaufstieg";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "Zeige ben\195\182tigte Monsterkills bis Levelaufstieg";
+L["TITAN_XP_MENU_RESET_SESSION"] = "Session Zur\195\188cksetzen";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "Z\195\164hler zur\195\188cksetzen";
+L["TITAN_XP_UPDATE_PENDING"] = "Aktualisiere...";
+L["TITAN_XP_KILLS_LABEL"] = "Monsterkills bis Levelaufstieg (bei %s XP Durchschnitt): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "Ca. Monsterkills: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "Sitzungszeit: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "Zeige Sitzungszeit";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*) stirbt, Ihr bekommt (%d+) Erfahrung.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "Ca. XP-Boni: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "XP-Boni bis Levelaufstieg (bei %s XP zuletzt): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "Zeige ungef\195\164hre XP Boni bis zum Levelaufstieg";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "Reparieren",
+	tooltip = "Haltbarkeits-Info",
+	button = "Haltbarkeit: ",
+	normal = "Reparaturkosten (Normal): ",
+	friendly = "Reparaturkosten (freundlich): ",
+	honored = "Reparaturkosten (wohlwollend): ",
+	revered = "Reparaturkosten (respektvoll): ",
+	exalted = "Reparaturkosten (ehrf\195\188rchtig): ",
+	buttonNormal = "Zeige normal",
+	buttonFriendly = "Zeige freundlich (5%)",
+	buttonHonored = "Zeige wohlwollend (10%)",
+	buttonRevered = "Zeige respektvoll (15%)",
+	buttonExalted = "Zeige ehrf\195\188rchtig (20%)",
+	percentage = "Prozent anzeigen",
+	itemnames = "Zeige Gegenstandsnamen an",
+	mostdamaged = "Zeige die am meisten besch\195\164digten an",
+	showdurabilityframe = "Zeige Haltbarkeits-Inforahmen",
+	undamaged = "Unbesch\195\164digte Gegenst\195\164nde anzeigen",
+	discount = "Rabatt",
+	nothing = "Nichts besch\195\164digt",
+	confirmation = "Wollt Ihr alles reparieren lassen?",
+	badmerchant = "Dieser H/195/164/ndler kann nicht reparieren. Es werden nur normale Reparaturkosten angezeigt.",
+	popup = "Zeige Reparieren-Popup",
+	showinventory = "Kalkuliere Inventar-Besch\195\164digung",
+	WholeScanInProgress = "Aktualisiere...",
+	AutoReplabel = "Automatische Reparatur",
+	AutoRepitemlabel = "Automatische Reparatur aller Gegenst\195\164nde",
+	ShowRepairCost = "Zeige Reparaturkosten",
+	ignoreThrown = "Ignoriere Wurfwaffen",
+	ShowItems = "Zeige Gegenst\195\164nde an",
+	ShowDiscounts = "Zeige Rabatte an",
+	ShowCosts = "Zeige Kosten an",
+	Items = "Gegenst\195\164nde",
+	Discounts = "Rabatte",
+	Costs = "Kosten",
+	CostTotal = "Gesamtkosten",
+	CostBag = "Taschenbasierte Kosten",
+	CostEquip = "Kosten f\195\188r angelegte Gegenst\195\164nde",
+	TooltipOptions = "Tooltip",
+};
+L["TITAN_REPAIR"] = "Titan Repair"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "Gildenbank-Guthaben:"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "Gildenbank-Guthabenbezug erlaubt :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "Benutze Gildenbank-Guthaben"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "Es ist nicht gen\195\188gend Guthaben auf der Gildenbank verf\195\188gbar oder du darfst nicht soviel beziehen."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "Du bist entweder nicht in einer Gilde oder hast keine Berechtigung, die Gildenbank für Reparaturkosten zu nutzen."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "Du kannst es Dir momentan nicht leisten, deine Gegenst\195\164nde zu reparieren."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "Schreibe Reparaturkosten in den Chat"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "Die Reparaturkosten betrugen "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "Gesamtes Gold auf";
+L["TITAN_GOLD_ITEMNAME"] = "Titan Gold";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "L\195\182sche Datenbank";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "Sitzung zur\195\188cksetzen";
+L["TITAN_GOLD_DB_CLEARED"] = "Titan Gold - Datenbank gel\195\182scht.";
+L["TITAN_GOLD_SESSION_RESET"] = "Titan Gold - Sitzung zur\195\188ck gesetzt.";
+L["TITAN_GOLD_MENU_TEXT"] = "Gold";
+L["TITAN_GOLD_TOOLTIP"] = "Gold-Info";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "Zeige Spielercharakter-Gold";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "Zeige Gold aller Spielercharaktere auf diesem Server";
+L["TITAN_GOLD_SESS_EARNED"] = "Eingenommen in der Sitzung";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "Eingenommen pro Stunde";
+L["TITAN_GOLD_SESS_LOST"] = "Ausgegeben in der Sitzung";
+L["TITAN_GOLD_PERHOUR_LOST"] = "Ausgegeben pro Stunde";
+L["TITAN_GOLD_STATS_TITLE"] = "Sitzungsstatistik";
+L["TITAN_GOLD_TTL_GOLD"] = "Gesamtes Gold";
+L["TITAN_GOLD_START_GOLD"] = "Anf\195\164ngliches Gold";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "Sortiere Tabelle nach Gold";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "Sortiere Tabelle nach Name";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "Zeige Gold pro Stunde";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "Verberge Gold pro Stunde";
+L["TITAN_GOLD_GOLD"] = "G";
+L["TITAN_GOLD_SILVER"] = "S";
+L["TITAN_GOLD_COPPER"] = "K";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "Sichtbar";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "Verborgen";
+L["TITAN_GOLD_DELETE_PLAYER"] = "Spielercharaktergoldinfo l\195\182schen";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Zeige Spielercharaktergoldinfo";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "Allianz";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "Horde";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."Warnung: "
+..FONT_COLOR_CODE_CLOSE.."Diese Einstellung l\195\182scht die TitanGold-Datenbank. "
+.."Wenn Du diese Aktion durchf\195\188hren willst, klicke auf 'Best\195\164tigen', anderenfalls auf 'Abbrechen' oder dr\195\188cke die ESC-Taste.";
+L["TITAN_GOLD_COIN_NONE"] = "Zeige keine Labels";
+L["TITAN_GOLD_COIN_LABELS"] = "Zeige Text Labels";
+L["TITAN_GOLD_COIN_ICONS"] = "Zeige Icon Labels";
+L["TITAN_GOLD_ONLY"] = "Zeige nur Gold";
+L["TITAN_GOLD_COLORS"] = "Zeige Gold (und Bruchteil)-Farben";
+L["TITAN_GOLD_MERGE"] = "Zusammengelegte Servers";
+L["TITAN_GOLD_SEPARATE"] = "Getrennte Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "Lautst\195\164rke einstellen";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "Momentane Gesamtlautst\195\164rke: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "Momentane Effektlautst\195\164rke: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "Momentane Umgebungslautst\195\164rke: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "Momentane Musiklautst\195\164rke: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "Momentane Mikrofonlautst\195\164rke: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "Momentane Lautsprecherlaust\195\164rke: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "Hinweis: Links-Klick um die";
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "Lautst\195\164rken anzupassen.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "Lautst\195\164rke: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "Lautst\195\164rke";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "Gesamt";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "Effekte";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "Umgebung";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "Musik";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "Mikrofon";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "Lautsprecher";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "Laut";
+L["TITAN_VOLUME_CONTROL_LOW"] = "Leise";
+L["TITAN_VOLUME_MENU_TEXT"] = "Lautst\195\164rkeregler";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "Zeige Audio-Optionen des Spiels an" ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "Ignoriere Die spieleseitigen Lautst\195\164rkeeinstellungen";
diff --git a/Titan/locale/Localization.ES.lua b/Titan/locale/Localization.ES.lua
new file mode 100644
index 0000000..b96df33
--- /dev/null
+++ b/Titan/locale/Localization.ES.lua
@@ -0,0 +1,570 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","esES")
+if not L then return end
+
+L["TITAN_DEBUG"] = "<Titan>";
+local TITAN_PANEL = "Titan Panel";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/D";
+L["TITAN_SECONDS"] = "segundos";
+L["TITAN_MINUTES"] = "minutos";
+L["TITAN_HOURS"] = "horas";
+L["TITAN_DAYS"] = "días";
+L["TITAN_SECONDS_ABBR"] = "s";
+L["TITAN_MINUTES_ABBR"] = "m";
+L["TITAN_HOURS_ABBR"] = "h";
+L["TITAN_DAYS_ABBR"] = "d";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "KB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kbps"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "Nada";
+L["TITAN_USE_COMMA"] = "Usar coma como separador numérico";
+L["TITAN_USE_PERIOD"] = "Use punto como separador numérico";
+
+--L["TITAN_MOVABLE_TOOLTIP"] = "Pincha para mover";
+
+L["TITAN_PANEL_ERROR_DUP_PLUGIN"] = " aparece duplicado. Esto puede hacer que Titan Panel falle, por favor, corrija el problema."
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "No puedes borrar el perfil actual.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."Aviso : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "Si deseas continuar con esta operación, pulsa 'Aceptar' (se recargará tu interface), de lo contrario pulsa 'Cancelar' o la tecla 'Escape'."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+.."Esta opción reseteará la configuración de tus barras y panel a los valores por defecto, y recreará tu perfil actual."
+..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+.."Esto recargará "..TITAN_PANEL.."."
+..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = "Sobre los plugins activados en "..TITAN_PANEL
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Información"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "Los siguientes plugins han solicitado ser activados en Titan.\n"
+.."Por favor envía cualquier incidencia que ocurra al autor del plugin."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Tipo"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Categoría"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Nombre del botón"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Estatus"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Incidencia"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Notas"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Indice"
+L["TITAN_PANEL_EXTRAS"] = "Extras de "..TITAN_PANEL
+L["TITAN_PANEL_EXTRAS_SHORT"] = "Extras"
+L["TITAN_PANEL_EXTRAS_DESC"] = "Estos plugins tienen cambios en su configuración que aún no han sido cargados.\n"
+.."Nota: Has de desconectar del juego para que la lista se actualice."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "Borar cambios de configuración"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "eliminados los cambios de configuración."
+L["TITAN_PANEL_CHARS"] = "Personajes"
+L["TITAN_PANEL_CHARS_DESC"] = "Estos personajes tienen cambios en su configuración."
+L["TITAN_PANEL_CHARS_DELETE_BUTTON"] = "Borrar datos del persojane"
+L["TITAN_PANEL_CHARS_DELETE_MSG"] = "eliminados los datos del personaje."
+L["TITAN_PANEL_REGISTER_START"] = "Registrar plugins de "..TITAN_PANEL.."..."
+L["TITAN_PANEL_REGISTER_END"] = "Proceso de registro completado."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffffResetea "..TITAN_PANEL.." a sus valores y posiciones por defecto.";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffResetea la escala de la fuente en los tooltips a su valor por defecto.";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffResetea la transparencia de los tooltips a su valor por defecto.";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffResetea la escala de "..TITAN_PANEL.." a su valor por defecto.";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffResetea el espaciado de los botones de "..TITAN_PANEL.." a su valor por defecto.";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffffAbre el interface de control para "..TITAN_PANEL..".";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffffAbre el interface de control de transarencia.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffffAbre el interface de control de Skin.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <nombre> <servidor>: |cffffffffActiva el perfil grabado.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<nombre>: |cffffffffpuede ser tanto un nombre de personaje como el nombre de un perfil personalizado."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<servidor>: |cffffffffpuede ser el nombre del servidor o 'TitanCustomProfile'."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffOrdena a "..TITAN_PANEL.." que cargue silenciosamente.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffOrdena a "..TITAN_PANEL.." que oculte la barra de comandos de la Sala de Orden.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE.."La escala de la fuente en los tooltips de Titan Panel ha sido reseteada.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE.."La transparencia de los tooltips de Titan Panel ha sido reseteada.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE.."La escala de Titan Panel ha sido reseteada.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE.."El espaciado de los botones de Titan Panel ha sido reseteado.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Global";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Perfil Global";
+L["TITAN_PANEL_GLOBAL_USE"] = "Usar Perfil Global";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Usar como Perfil Global";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Utiliza un perfil global para todos los personajes";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "restableciendo opciones";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "No se puede cargar un perfil cuando el global está en uso";
+
+-- general panel locale
+L["TITAN_PANEL"] = "Titan Panel";
+L["TITAN_PANEL_VERSION_INFO"] = "|cffffd700 por el equipo de desarrollo de  |cffff8c00"..TITAN_PANEL.."";
+L["TITAN_PANEL_MENU_TITLE"] = TITAN_PANEL;
+L["TITAN_PANEL_MENU_HIDE"] = "Ocultar";
+L["TITAN_PANEL_MENU_CUSTOMIZE"] = "Personalizar";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(En combate)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(Recargar UI)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "Mostrar texto en color";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "Mostrar icono";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "Mostrar etiqueta de texto";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "Auto-ocultar";
+L["TITAN_PANEL_MENU_BGMINIMAP"] = "Campo de batalla en mini-mapa";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "Centrar texto";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Mostrar barra superior";
+L["TITAN_PANEL_MENU_DISPLAY_BOTH"] = "Mostrar ambas barras";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "Desactivar ajuste de pantalla";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "Desactivar ajuste de minimapa";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "Ajuste automático de log";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "Ajuste automático de bolsas";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "Ajuste automático de la ventana de tickets";
+L["TITAN_PANEL_MENU_BUILTINS"] = "De serie en Titan";
+L["TITAN_PANEL_MENU_LEFT_SIDE"] = "Lado izquierdo";
+L["TITAN_PANEL_MENU_RIGHT_SIDE"] = "Lado derecho";
+L["TITAN_PANEL_MENU_PROFILES"] = "Perfiles";
+L["TITAN_PANEL_MENU_PROFILE"] = "Perfil ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "Personalizado";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " ha sido borrado.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "Servidor";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "Personaje";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "Se recargará tu interface en cuanto pulses 'Ok' para poder grabar tu perfil personalizado.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "Introduce un nombre para tu perfil personalizado::\n(20 carácteres máximo, sin espacios, diferencia entre mayúsculas y minúsculas)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "La configuración actual del panel se va a grabar en el perfil llamado: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "El nombre de perfil introducido ya existe. ¿Estás seguro de que quieres sobreescribirlo? Pulsa 'Aceptar' si lo deseas, de lo contrario pulsa 'Cancelar' o la tecla 'Escape'.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "Gestionar";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "Cargar ajustes";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "Borrar";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "Grabar";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "Configuración";
+L["TITAN_PANEL_OPTIONS"] = "Opciones";
+L["TITAN_PANEL_MENU_TOP"] = "Superior"
+L["TITAN_PANEL_MENU_TOP2"] = "Superior 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "Inferior"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "Inferior 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = "Tooltips y Marcos de "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Tooltips y Marcos";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Barras superiores";
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Barras inferiores";
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "Barras";
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = "Barras superiores "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = "Barras inferiores "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Tooltips";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "Marcos";
+L["TITAN_PANEL_MENU_PLUGINS"] = "Plugins";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "Bloquear botones";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "Mostrar versiones del plugin";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "Plugin del lado derecho";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "Forzar lanzadores LDB en el lado derecho";
+L["TITAN_PANEL_MENU_DISABLE_FONT"] = "Desactivar escalado de fuentes";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"De serie en Titan","General","Combate","Información","Interfaz","Profesión"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "Mostrar tooltips";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "Ocultar tooltips en combate";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Ocultar las barras en combate"
+L["TITAN_PANEL_MENU_RESET"] = "Volver a los ajustes por defecto de "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "Skins";
+L["TITAN_PANEL_MENU_FONT"] = "Fuente";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "Fuente del Panel"
+L["TITAN_PANEL_MENU_ENABLED"] = "Activado";
+L["TITAN_PANEL_MENU_DISABLED"] = "Desactivado";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Mayus-Izdo.";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Mayus-Dcho.";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Mostrar texto del plugin";
+L["TITAN_PANEL_MENU_LDB_TREAT"] = "Tratar como origen de datos";
+L["TITAN_PANEL_MENU_LDB_SLAP"] = "Si necesitas usar esto solicítale al autor que cambie el tipo LDB";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Siempre activo";
+L["TITAN_PANEL_MENU_POSITION"] = "Posición";
+L["TITAN_PANEL_MENU_BAR"] = "Barra";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Elige en qué barra se mostrará el plugin";
+L["TITAN_PANEL_MENU_SHOW"] = "Mostrar plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Actualizar plugins";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Actualizar texto y posición de plugin";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Carga silenciosa";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Ocultar la barra de comandos de la Sala de Orden";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Versión";
+L["TITAN_ABOUT_AUTHOR"] = "Autor";
+L["TITAN_ABOUT_CREDITS"] = "Créditos";
+L["TITAN_ABOUT_CATEGORY"] = "Categoría";
+L["TITAN_ABOUT_EMAIL"] = "Email";
+L["TITAN_ABOUT_WEB"] = "Sitio Web";
+L["TITAN_ABOUT_LICENSE"] = "Licencia";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "Addon de la barra de información. Permite a los usuarios añadir orígenes de datos o plugins lanzadores en un panel de control situado en la parte superior o inferior de la pantalla.";
+L["TITAN_TRANS_MENU_TEXT"] = TITAN_PANEL.." Transparencia";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "Transparencia";
+L["TITAN_TRANS_MENU_DESC"] = "Ajustar transparencia para las barras "..TITAN_PANEL.." y sus tooltips.";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "Barra Principal";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "Barra Auxiliar";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "Tooltips";
+L["TITAN_TRANS_MAIN_BAR_DESC"] = "Establece la transparencia para la Barra Principal.";
+L["TITAN_TRANS_AUX_BAR_DESC"] = "Establece la transparencia para la Barra Auxiliar (inferior).";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "Establece la transparencia para los tooltips de varios plugins.";
+L["TITAN_UISCALE_MENU_TEXT"] = "Escala y Fuentes de "..TITAN_PANEL;
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Escala y Fuentes";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "Escala del interface";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "Escala de "..TITAN_PANEL;
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "Espaciado de los botones";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Espaciado de iconos";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "Tamaño de la fuente de los tooltips";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "Desactivar tamaño de la fuente de los tooltips";
+L["TITAN_UISCALE_MENU_DESC"] = "Controla varios aspectos del interface y "..TITAN_PANEL..".";
+L["TITAN_UISCALE_SLIDER_DESC"] = "Establece la escala de todo el interface.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "Establece la escala para los distintos botones e iconos de "..TITAN_PANEL..".";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "Ajusta el espacio entre los plugins del lado izquierdo.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Ajusta el espacio entre los iconos de los plugins del lado derecho.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "Ajusta la escala de los tooltips de varios plugins.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "Desactiva el control de la escala de los tooltips en "..TITAN_PANEL..".";
+L["TITAN_SKINS_TITLE"] = "Skins de "..TITAN_PANEL;
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Skins - Personalizado";
+L["TITAN_SKINS_TITLE_CUSTOM"] = "Skins Personalizados de "..TITAN_PANEL;
+L["TITAN_SKINS_MAIN_DESC"] = "Se asume que todos los skins personalizados están en: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ ".."\n"
+			.."\n"..TITAN_PANEL.." y los skins personalizados son almacenados en el directorio Custom."
+L["TITAN_SKINS_LIST_TITLE"] = "Lista de skins";
+L["TITAN_SKINS_SET_DESC"] = "Elige que skin quieres para las barras "..TITAN_PANEL..".";
+L["TITAN_SKINS_SET_HEADER"] = "Establece skin de "..TITAN_PANEL;
+L["TITAN_SKINS_RESET_HEADER"] = "Resetea el skin de "..TITAN_PANEL;
+L["TITAN_SKINS_NEW_HEADER"] = "Añadir nuevo skin";
+L["TITAN_SKINS_NAME_TITLE"] = "Nombre de skin";
+L["TITAN_SKINS_NAME_DESC"] = "Introduce un nombre para tu nuevo skin. Se usará en la lista desplegable de skins.";
+L["TITAN_SKINS_NAME_EXAMPLE"] = "ejemplo: mi skin Titan";
+L["TITAN_SKINS_PATH_TITLE"] = "<Ruta del skin>";
+L["TITAN_SKINS_PATH_DESC"] = "<Ruta del skin> en el directorio de instalación de "..TITAN_PANEL..". Mirar el ejemplo superior."
+L["TITAN_SKINS_ADD_HEADER"] = "Añadir skin";
+L["TITAN_SKINS_ADD_DESC"] = "Añade un nuevo skin a la lista de skins disponibles para el Panel.";
+L["TITAN_SKINS_REMOVE_HEADER"] = "Eliminar skin";
+L["TITAN_SKINS_REMOVE_DESC"] = "Elige un skin para eliminar de la lista de skins disponibles para el Panel.";
+L["TITAN_SKINS_REMOVE_BUTTON"] = "Eliminar";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "Elimina el skin selecionado de la lista de skins disponibles para el Panel.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "Serás el único responsable si eliminas cualquier skin personalizado no deseado "
+	.." del directorio de instalación de "..TITAN_PANEL..". Los addons no pueden añadir o eliminar archivos."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "Resetear lista";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "Resetea la lista de skins a sus valores por defecto, los skins incluidos de serie.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "Elige el tipo de letra utilizado por los diferentes plugins en las barras "..TITAN_PANEL..".";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "Tamaño de la fuente";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "Establece el tamaño para el tipo de letra de "..TITAN_PANEL..".";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = "Frame Strata de "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "Establece la prioridad de visionado de las Barras "..TITAN_PANEL..".";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Avanzado";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Cambia el retraso tras eventos sólo si tienes problemas de desajuste con los marcos.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Entrando en el reino";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Cambia el valor (habitualmente incrementándolo) si los marcos no se ajustan al entrar o salir del reino o de una instancia.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Vehículo";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Cambia el valor (habitualmente incrementándolo) si los marcos no se ajustan al entrar o salir de un vehículo.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "Activar/Desactivar la característica de auto-ocultación de "..TITAN_PANEL;
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "Bolsas: ";
+L["TITAN_BAG_TOOLTIP"] = "Uso de bolsas";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "Consejo: Click-Izq. Abre todas las bolsas.";
+L["TITAN_BAG_MENU_TEXT"] = "Bolsas";
+L["TITAN_BAG_USED_SLOTS"] = "Huecos usados";
+L["TITAN_BAG_FREE_SLOTS"] = "Huecos libres";
+L["TITAN_BAG_BACKPACK"] = "Mochila";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "Mostrar huecos usados";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "Mostrar huecos libres";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "Mostrar tooltip detallado";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "Ignorar bolsas";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "Ignorar bolsas de profesión";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Faltriquera de tejido mágico encantada","Bolsa de paño rúnico encantada","Cartera de encantador","Bolsa de encantamiento grande","Bolsa de fuego de hechizo","Bolsa misteriosa","Bolsa de otro mundo","Bolso de noche encantador \"Trasportín exclusivo\""};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Caja de herramientas pesada","Caja de herramientas de hierro vil","Caja de herramientas de titanio","Caja de herramientas de elementium","Bolsa de alta tecnología \"Transportín Maddy\""};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Faltriquera de hierbas","Bolsa de hierbas Cenarion","Cartera de Cenarius","Bolsa de expedición de Hyjal","Bolsa esmeralda","Bolsa Botánica de Mycah","Portahierbas \"Trasportín ecológico\""};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Cartera de escriba","Bolsa de bolsillos infinitos","Cartera de estudiante \"Transportín Xandera\"", "Bolsa pulida de inscripción"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Faltriquera de gemas","Bolsa de joyas","Agarre con tachuelas de gemas \"Trasportín exclusivo\""};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Cartera de peletero","Mochila de viaje de trampero","Bolsa de muchos pellejos","Bolsa de cuero \"Transportín Meeya\"", "Bolsa pulida de piel"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Saco de minero","Bolsa de minero reforzada","Bolsa de minero descomunal","Bolsa de metal precioso \"Trasportín Christina\"", "Bolsa pulida de minería"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Caja de aparejos \"Maestro del Cebo\""};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Refrigerador portátil", "Unidad de refrigeración avanzada"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "Reloj";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "Ajuste horario: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "Hora local: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "Hora del servidor: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "Hora del servidor ajustada: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "Consejo: Click-Izq. ajusta el horario"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "y el formato de tiempo 12/24H.";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Sift + Click-Izq. para activar y desactivar el calendario.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "Ajuste horario: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "Ajuste horario";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24H";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "Alterna la hora mostrada entre los formatos de 12 horas y 24 horas";
+L["TITAN_CLOCK_MENU_TEXT"] = "Reloj";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "Muestra hora local (L)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "Muestra hora del servidor (S)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "Muestra hora del servidor ajustada (A)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "Mostrar al final a la derecha";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "Ocultar botón de hora/calendario";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Ocultar botón de hora";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Ocultar botón de calendario";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "Formato de coordenadas";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "Loc: ";
+L["TITAN_LOCATION_TOOLTIP"] = "Ubicación";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "Consejo: Shift + Click-Izq. para poner la ubicación";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "en el chat de mensajes.";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "Zona: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "Subzona: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "Info PVP: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "Ubicación de mi casa";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "Posada: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "Ubicación";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "Mostrar texto de zona";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "Mostrar coordenadas en el mapamundi";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "Cursor(X,Y): %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "Jugador(X,Y): %s";
+L["TITAN_LOCATION_NO_COORDS"] = "Sin Coordenadas";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "Mostrar el nombre de la ubicación sobre el minimapa";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Actualizar mapamundi cuando cambia la zona";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "FPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "FPS actuales: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "FPS de nedia: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "FPS mínimos: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "FPS máximos: ";
+L["TITAN_FPS_TOOLTIP"] = "Imágenes por segundo";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "Latencia: ";
+L["TITAN_LATENCY_TOOLTIP"] = "Estado de red";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Latencia del reino (hogar): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Latencia del juego (mundo): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "Ancho de banda entrante: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "Ancho de banda de salida: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Latencia";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "Saqueo: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "Libre para todos";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "Por turnos";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "Maestro despojador";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "Botín de grupo";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "Necesidad antes que codicia";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "Tipo de saqueo";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "Tipo de saqueo";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "Tirar dados";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "Consejo: Click-Izq. para tirar dados.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "Seleccione el valor de la tirada de dados con el menu de Click-Dcho.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "Dificultad de Mazmorra";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "Dificultad de Banda";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "Mostrar dificultad de Mazmorra/Banda";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "Establecer dificultad de Mazmorra";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "Establecer dificultad de Banda";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "Auto (según grupo)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "Memoria: ";
+L["TITAN_MEMORY_TOOLTIP"] = "Uso de Memoria";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "Actual: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "Inicial: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "Ratio de Incremento: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "Monitorización";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "Monitorización";
+L["TITAN_PERFORMANCE_ADDONS"] = "Addon de uso";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "Addon de uso de memoria";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "Formato de memoria";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "Addon de uso de procesador";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "Nombre:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "Uso";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "Ratio";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "Uso de memoria total:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "Uso de procesador total:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "Mostrar FPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "Mostrar latencia del reino (hogar)";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Mostrar latencia del juego (mundo)";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "Mostrar nemoria";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "Mostrar Addon de uso de memoria";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "Mostrar Addon de ratio de uso";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "Modo de monitorización del procesador";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "Activar monitorización del procesador ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "Desactivar monitorización del procesador ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "Addons monitorizados: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "Addons monitorizados";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "Consejo: Click-Izq. para forzar una limpieza de memoria.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "XP/hr este nivel: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "XP/hr esta sesión: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "Tiempo para subir: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "Nivel completo: ";
+L["TITAN_XP_TOTAL_RESTED"] = "XP bonifcada por descanso: ";
+L["TITAN_XP_XPTOLEVELUP"] = "XP para subir: ";
+L["TITAN_XP_TOOLTIP"] = "INFO XP";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "Tiempo total jugado: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "Tiempo jugado este nivel: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "Tiempo jugado esta sesión: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "XP requerida este nivel: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "XP ganada este nivel: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "XP necesaria este nivel: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "XP ganada esta sesión: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "XP/hr este nivel: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "XP/hr esta sesión: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "Tiempo para subir (ratio de nivel): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "Tiempo para subir (ratio de sesión): ";
+L["TITAN_XP_MENU_TEXT"] = "XP";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "Mostrar XP/hr este nivel";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "Mostrar XP/hr esta sesión";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Mostrar descanso/XP para subir";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "Botón";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "Mostrar bonus XP por descanso";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "Mostrar XP para subir";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "Mostrar muertes estimadas para subir";
+L["TITAN_XP_MENU_RESET_SESSION"] = "Reiniciar sesión";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "Actualizar temporizadores";
+L["TITAN_XP_UPDATE_PENDING"] = "Actualizando...";
+L["TITAN_XP_KILLS_LABEL"] = "Muertes para subir (a %s XP ganados la última vez): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "Muertes estimadas: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "Tiempo en esta sesión: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "Mostrar tiempo en esta sesión";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*) muere, ganas (%d+) de experiencia.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "Gananc. Est.: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "Ganancias de XP para subir (a %s XP ganados la útltima vez): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "Mostrar ganancias de XP estimadas para subir";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "Reparación",
+	tooltip = "Información de reparación",
+	button = "Durabilidad: ",
+	normal = "Coste de reparación (Neutral): ",
+	friendly = "Coste de reparación (Amistoso): ",
+	honored = "Coste de reparación (Honorable): ",
+	revered = "Coste de reparación (Venerado): ",
+	exalted = "Coste de reparación (Exaltado): ",
+	buttonNormal = "Mostrar neutral",
+	buttonFriendly = "Mostrar amistoso (5%)",
+	buttonHonored = "Mostrar honorable (10%)",
+	buttonRevered = "Mostrar venerado (15%)",
+	buttonExalted = "Mostrar exaltado (20%)",
+	percentage = "Mostrar como porcentaje",
+	itemnames = "Mostrar nombres de objetos",
+	mostdamaged = "Mostrar el más dañado",
+	showdurabilityframe = "Mostrar tooltip informativo",
+	undamaged = "Mostrar objetos sin dañar",
+	discount = "Mostrar descuento y objetos",
+	nothing = "Nada dañado",
+	confirmation = "¿Quieres reparar todos los objetos equipados?",
+	badmerchant = "Este comerciante no puede reparar. Mostrando entonces los costes de reparación normales.",
+	popup = "Mostrar ventana emergente de reparación",
+	showinventory = "Calcular daño de inventario",
+	WholeScanInProgress = "Escaneando...",
+	AutoReplabel = "Auto-reparar",
+	AutoRepitemlabel = "Auto reparar todos los objetos",
+	ShowRepairCost = "Mostrar coste de reparar",
+	ignoreThrown = "Ignorar armas arrojadizas",
+	ShowItems = "Mostrar objetos",
+	ShowDiscounts = "Mostrar descuentos",
+	ShowCosts = "Mostrar costes",
+	Items = "Objetos",
+	Discounts = "Descuentos",
+	Costs = "Costes",
+	TooltipOptions = "Tooltip",
+	CostTotal = "Coste total",
+	CostBag = "Coste de bolsas",
+	CostEquip = "Coste del equipo",
+};
+
+L["TITAN_REPAIR"] = "Titan Repair"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "Efectivo en el banco de la Hermandad: "
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "Retirada de efectivo de la Hermandad permitida: "
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "Usar efectivo del banco de la Hermandad"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "El banco de la Hermandad no puede permitirse el coste de la reparación, o no puedes retirar tanto dinero."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "O no perteneces a una Hermandad, o no tienes permiso para usar el banco de la Hermandad para reparar tus objetos."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "No puedes permitirte reparar, en este momento."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "Informar en el chat del coste de la reparación"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "El coste de la reparación ha sido "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "Oro total en";
+L["TITAN_GOLD_ITEMNAME"] = "Titan Gold";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "Limpiar base de datos";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "Resetear sesión actual";
+L["TITAN_GOLD_DB_CLEARED"] = "Titan Gold - Base de datos limpiada.";
+L["TITAN_GOLD_SESSION_RESET"] = "Titan Gold - Sesión reiniciada.";
+L["TITAN_GOLD_MENU_TEXT"] = "Oro";
+L["TITAN_GOLD_TOOLTIP"] = "Información de Oro";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "Mostrar oro del jugador";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "Mostrar oro en el servidor";
+L["TITAN_GOLD_SESS_EARNED"] = "Ganado esta sesión";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "Ganado por hora";
+L["TITAN_GOLD_SESS_LOST"] = "Perdido esta sesión";
+L["TITAN_GOLD_PERHOUR_LOST"] = "Perdido por hora";
+L["TITAN_GOLD_STATS_TITLE"] = "Estadisticas de esta sesión";
+L["TITAN_GOLD_TTL_GOLD"] = "Oro total";
+L["TITAN_GOLD_START_GOLD"] = "Oro al principio";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "Ordenar tabla por oro";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "Ordenar tabla por nombre";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "Mostrar oro por hora";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "Ocultar oro por hora";
+L["TITAN_GOLD_GOLD"] = "o";
+L["TITAN_GOLD_SILVER"] = "p";
+L["TITAN_GOLD_COPPER"] = "c";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "visible";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "oculto";
+L["TITAN_GOLD_DELETE_PLAYER"] = "Eliminar personaje";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Mostrar personaje";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "Alianza";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "Horda";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."Cuidado: "
+..FONT_COLOR_CODE_CLOSE.."Esta opción purgará los datos de Titan Gold. "
+.."Si deseas seguir con esta operación, pulsa 'Aceptar', de lo contrario pulsa 'Cancelar' o la tecla 'Escape'.";
+L["TITAN_GOLD_COIN_NONE"] = "No mostrar etiquetas";
+L["TITAN_GOLD_COIN_LABELS"] = "Mostrar etiquetas en texto";
+L["TITAN_GOLD_COIN_ICONS"] = "Mostrar etiquetas en texto";
+L["TITAN_GOLD_ONLY"] = "Mostrar sólo oro";
+L["TITAN_GOLD_COLORS"] = "Mostrar colores en el oro";
+L["TITAN_GOLD_MERGE"] = "Unificar servidores";
+L["TITAN_GOLD_SEPARATE"] = "Separar servidores";
+
+L["TITAN_VOLUME_TOOLTIP"] = "Control de Volumen";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "Volumen general: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "Volumen de los efectos: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "Volumen de ambiente: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Volumen de los diálogos: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "Volumen de la música: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "Volumen del micrófono: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "Volumen de los altavoces: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "Consejo: Click-Izq. para ajustar"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "el volumen.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "Volumen: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "Volumen";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "General";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "Efectos";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "Ambiente";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Diálogos";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "Música";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "Micrófono";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "Altavoz";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "Alto";
+L["TITAN_VOLUME_CONTROL_LOW"] = "Bajo";
+L["TITAN_VOLUME_MENU_TEXT"] = "Ajuste de Volumen";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "Mostrar opciones de sonido/voz";
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "Invalida la configuración de volumen de Blizzard";
diff --git a/Titan/locale/Localization.FR.lua b/Titan/locale/Localization.FR.lua
new file mode 100644
index 0000000..518ab27
--- /dev/null
+++ b/Titan/locale/Localization.FR.lua
@@ -0,0 +1,553 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","frFR")
+if not L then return end
+
+L["TITAN_PANEL"] = "Titan Panel";
+local TITAN_PANEL = "Titan Panel";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/A";
+L["TITAN_SECONDS"] = "secondes";
+L["TITAN_MINUTES"] = "minutes";
+L["TITAN_HOURS"] = "heures";
+L["TITAN_DAYS"] = "jours";
+L["TITAN_SECONDS_ABBR"] = "s";
+L["TITAN_MINUTES_ABBR"] = "m";
+L["TITAN_HOURS_ABBR"] = "h";
+L["TITAN_DAYS_ABBR"] = "d";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "Ko/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kb/s"
+L["TITAN_MEGABYTE"] = "Mo";
+L["TITAN_NONE"] = "Aucun";
+L["TITAN_USE_COMMA"] = "Utiliser une virgule pour le séparateur de milliers";
+L["TITAN_USE_PERIOD"] = "Utiliser un point pour le séparateur de milliers";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "Vous ne pouvez pas supprimer votre profil en cours.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."Attention : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "Si vous voulez continuer cette opération, appuyez sur 'Accepter' (l'interface va se recharger), sinon appuyez sur 'Annuler' ou la touche 'Echap'."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."cela va réinitialiser la/les barres et tous les paramètres de "..TITAN_PANEL.." à leur valeur par défaut et va recréer votre profil. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."Cela va recharger "..TITAN_PANEL..". "
+	..TITAN_PANEL_RELOAD_TEXT
+
+L["TITAN_PANEL_ATTEMPTS"] = TITAN_PANEL.." - Tentatives d'enregistrement" -- ??
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Enregistrement" -- ??
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "Les plugins ci-dessous ont cherché à s'enregistrer auprès de "..TITAN_PANEL..".\n"
+	.."Tout problème rencontré avec ces plugins est à rapporter à l'auteur du plugin."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Type"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Catégorie"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Nom du bouton"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Statut"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Problème"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Notes"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Index dans la table"
+
+L["TITAN_PANEL_EXTRAS"] = TITAN_PANEL.." Extras"
+L["TITAN_PANEL_EXTRAS_DESC"] = "Plugins non lancés ayant des données de configuration.\n"
+	.."Note: Vous devez vous déconnecter pour que les suppressions soient prises en compte."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "Supprimer les données de configuration"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "configuration supprimée."
+
+L["TITAN_PANEL_CHARS"] = "Personnages"
+L["TITAN_PANEL_CHARS_DESC"] = "Ces personnages ont des données de configuration.\n"
+	.."Note: Vous devez vous déconnecter pour que les suppressions soient prises en compte."
+
+L["TITAN_PANEL_REGISTER_START"] = "Enregistrement des plugins "..TITAN_PANEL.."..."
+L["TITAN_PANEL_REGISTER_END"] = "Enregistrement des plugins effectué."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: | cffffffffRéinitialise les valeurs et positions de "..TITAN_PANEL..".";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffRéinitialise l'échelle des polices des tooltips de "..TITAN_PANEL..".";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffRéinitialise la transparence des tooltips de "..TITAN_PANEL..".";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffRéinitialise l'échelle de "..TITAN_PANEL..".";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffRéinitialise l'espacement des boutons de "..TITAN_PANEL..".";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffffOuvre le panneau de configuration de "..TITAN_PANEL..".";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffffOuvre le panneau de configuration de la transparence.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffffOuvre le panneau de configuration des skins.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <nom> <serveur>: |cffffffffcharge le profil choisi :";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<nom>: |cffffffffle nom du personnage ou le profil personnalisé."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<serveur>: |cffffffffle nom du serveur ou 'TitanCustomProfile'."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffToggles "..TITAN_PANEL.." pour charger silencieusement.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE.."L'échelle des polices des tooltips de "..TITAN_PANEL.." a été réinitialisée.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE.."La transparence des tooltips de "..TITAN_PANEL.." a été réinitialisée.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE.."L'échelle de "..TITAN_PANEL.." a été réinitialisée.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE.."L'espacement des boutons de "..TITAN_PANEL.." a été réinitialisé.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Global";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Profil global";
+L["TITAN_PANEL_GLOBAL_USE"] = "Utiliser le profil global";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Utiliser en tant que profil global";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Utiliser un profil global pour tous les persos";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "réinitialisation des options";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "Vous ne devriez pas charger un profil lorsqu'un profil global est en cours d'utilisation";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffffd700 par le |cffff8c00"..TITAN_PANEL.." Development Team";
+L["TITAN_PANEL_MENU_TITLE"] = TITAN_PANEL;
+L["TITAN_PANEL_MENU_HIDE"] = "Cacher";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(En Combat)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(Recharge l'interface)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "Colorer le texte";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "Montrer l'icône";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "Montrer le titre";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "Cacher automatiquement";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "Centrer le texte";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Afficher la barre";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "Ne pas ajuster l'écran";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "Ne pas ajuster la minicarte";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "Ajuster automatiquement la fenêtre de discussion";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "Ajuster automatiquement les sacs";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "Ajuster automatiquement le ticket MJ";
+L["TITAN_PANEL_MENU_PROFILES"] = "Profils";
+L["TITAN_PANEL_MENU_PROFILE"] = "Profil ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "Personnalisation";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " a été supprimé.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "Serveur";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "Personnage";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "L'interface va maintenant être recharcée lorsque vous appuierez sur 'OK' pour permettre d'enregistrer votre profil perso.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "Entrez un nom pour votre profil perso :\n(20 lettres max, pas d'espaces)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "Les paramètres actuels vont être enregistrés sous le profil : ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "Le nom du profil existe déjà. Veuillez saisir un autre nom.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "Gestion";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "Chargement";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "Suppression";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "Sauvegarde";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "Configuration";
+L["TITAN_PANEL_OPTIONS"] = "Options";
+L["TITAN_PANEL_MENU_TOP"] = "Haut 1"
+L["TITAN_PANEL_MENU_TOP2"] = "Haut 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "Bas 1"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "Bas 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = TITAN_PANEL.." Tooltips et Frames"; -- ??
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Tooltips et Frames"; -- ??
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Barres du haut" -- ??
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Barres du bas" -- ??
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "Barres";
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = TITAN_PANEL.." Barres du haut";
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = TITAN_PANEL.." Barres du bas";
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Tooltips";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "Frames";
+L["TITAN_PANEL_MENU_PLUGINS"] = "Plug-ins";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "Verrouiller les boutons";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "Afficher les versions";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "Plugin du côté droit";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "Forcer les launchers du côté droit";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"Interne","Généralités","Combat","Informations","Interface","Métiers"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "Afficher les conseils";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "Cacher les tooltips en combat";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Bloquer l'affichage automatique en combat";
+L["TITAN_PANEL_MENU_RESET"] = "Réinitialiser "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "Paramètres du skin";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "Police des panneaux"
+L["TITAN_PANEL_MENU_ENABLED"] = "Activé";
+L["TITAN_PANEL_MENU_DISABLED"] = "Désactivé";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Déplacer à gauche";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Déplacer à droite";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Afficher le texte";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Toujours sur la barre"; --??
+L["TITAN_PANEL_MENU_POSITION"] = "Position";
+L["TITAN_PANEL_MENU_BAR"] = "Barre";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Choisir la barre sur laquelle afficher le plugin :";
+L["TITAN_PANEL_MENU_SHOW"] = "Afficher le plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Rafraîchir les plugins";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Rafraîchit le texte et la position des plugins";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Chargement silencieux";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Version";
+L["TITAN_ABOUT_AUTHOR"] = "Auteur";
+L["TITAN_ABOUT_CREDITS"] = "Credits";
+L["TITAN_ABOUT_CATEGORY"] = "Catégorie";
+L["TITAN_ABOUT_EMAIL"] = "Email";
+L["TITAN_ABOUT_WEB"] = "Site web";
+L["TITAN_ABOUT_LICENSE"] = "Licence";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "Addon de barres d'informations. Permet aux utilisateurs d'ajouter des plugins sur les barres placées en haut et/ou en bas de l'écran.";
+L["TITAN_TRANS_MENU_TEXT"] = TITAN_PANEL.." Transparence";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "Transparence";
+L["TITAN_TRANS_MENU_DESC"] = "Ajuste la transparence pour les barres de "..TITAN_PANEL.." et les tooltips.";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "Barre principale";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "Barre secondaire";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "Tooltip";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "Ajuste la transparence pour les tooltips des divers plugins.";
+L["TITAN_UISCALE_MENU_TEXT"] = TITAN_PANEL.." Echelle et police";
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Echelle et police";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "Echelle de l'interface";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "Echelle de "..TITAN_PANEL;
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "Espacement des boutons";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Espacement des icônes";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "Echelle des polices des tooltips";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "Désactiver l'échelle de la police des tooltips";
+L["TITAN_UISCALE_MENU_DESC"] = "Paramètre l'aspect global de l'UI et de "..TITAN_PANEL..".";
+L["TITAN_UISCALE_SLIDER_DESC"] = "Ajuste l'échelle de l'interface entière.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "Ajuste l'échelle pour les boutons et icônes de "..TITAN_PANEL..".";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "Ajuste l'espace entre les plugins du côté gauche.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Ajuste l'espace entre les plugins du côté droit.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "Ajuste l'échelle des tooltips des divers plugins.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "Désactiver la mise à l'échelle de la police des tooltips de "..TITAN_PANEL..".";
+
+L["TITAN_SKINS_TITLE"] = TITAN_PANEL.." Skins";
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Skins persos";
+L["TITAN_SKINS_TITLE_CUSTOM"] = TITAN_PANEL.." Skins personnalisés";
+L["TITAN_SKINS_MAIN_DESC"] = "Tous les skins personnalisés doivent être dans : \n"
+	.."..\\Interface\\AddOns\\Titan\\Artwork\\Custom\\<Mon Skin>\\ ".."\n"
+	.."\nLes skins personnalisés et ceux de "..TITAN_PANEL.." sont stockés dans le dossier 'Custom'."
+L["TITAN_SKINS_LIST_TITLE"] = "Liste des skins";
+L["TITAN_SKINS_SET_DESC"] = "Sélectionne un skin pour les barres de "..TITAN_PANEL..".";
+L["TITAN_SKINS_SET_HEADER"] = "Choix du skin";
+L["TITAN_SKINS_RESET_HEADER"] = "Réinitialiser les skins de "..TITAN_PANEL;
+L["TITAN_SKINS_NEW_HEADER"] = "Ajout d'un nouveau skin";
+L["TITAN_SKINS_NAME_TITLE"] = "Nom du skin";
+L["TITAN_SKINS_NAME_DESC"] = "Entrez un nom pour votre nouveau skin. Il sera utilisé dans la liste des skins.";
+L["TITAN_SKINS_PATH_TITLE"] = "Chemin du skin";
+L["TITAN_SKINS_PATH_DESC"] = "Entrez le chemin exact où se trouve votre artwork, comme montré par l'exemple."; -- et expliqué dans les 'Notes'.";
+L["TITAN_SKINS_ADD_HEADER"] = "Ajouter le skin";
+L["TITAN_SKINS_ADD_DESC"] = "Ajouter un nouveau skin à la liste des skins disponibles.";
+L["TITAN_SKINS_REMOVE_HEADER"] = "Suppression d'un skin";
+L["TITAN_SKINS_REMOVE_DESC"] = "Supprime un skin de la liste des skins disponibles.";
+L["TITAN_SKINS_REMOVE_BUTTON"] = "Supprimer le skin";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "Supprime le skin sélectionné de la liste des skins disponibles.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "Vous êtes responsable de la suppression des skins personnalisés que vous ne souhaitez pas "
+	.."dans le dossier de "..TITAN_PANEL..". Les addons ne peuvent pas ajouter ou supprimer des fichiers."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "Réinit. la liste";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "Réinitialise la liste de skin aux skins internes de "..TITAN_PANEL..".";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "Sélectionne la police pour les divers plugins sur les barres de "..TITAN_PANEL..".";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "Taille de police";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "Sélectionne la taille des polices pour les divers plugins sur les barres de "..TITAN_PANEL..".";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = TITAN_PANEL.." Altitude des frames";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "Ajuste l'altitude des frames pour les barres de "..TITAN_PANEL..".";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Avancé";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Changez les timers uniquement si vous avez des problèmes avec l'ajustement des frames.".."\n"; -- ??
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Entrée dans le monde";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Utiliser (généralement en augmentant) si les frames ne s'ajustent pas lors de chargement du monde (ou d'une instance).";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Véhicule";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Utiliser (généralement en augmentant) si les frames ne s'ajustent pas lorsqu'on entre / descend d'un véhicule.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "Changer l'affichage automatique du panneau on/off";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "Sacs : ";
+L["TITAN_BAG_TOOLTIP"] = "Utilisation des sacs";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "Info : clic-gauche pour ouvrir tous les sacs.";
+L["TITAN_BAG_MENU_TEXT"] = "Sac";
+L["TITAN_BAG_USED_SLOTS"] = "Emplacements utilisés";
+L["TITAN_BAG_FREE_SLOTS"] = "Emplacements libres";
+L["TITAN_BAG_BACKPACK"] = "Sac à dos";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "Montrer les emplacements utilisés";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "Montrer les emplacements libres";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "Afficher les détails";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "Ignorer les sacs de métiers";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "Ignorer les emplacements des sacs spécifiques aux métiers";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Bourse enchantée en tisse-mage","Sac de soirée enchanteur en exclusivité pour La Tannée","Sac enchanté en étoffe runique","Sacoche d'enchanteur","Grand sac d'enchantement","Sac du feu-sorcier","Sac mystérieux","Sac surnaturel"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Sac de haute technologie « Maddy » par La Tannée","Lourde boîte à outils","Boîte à outils en gangrefer","Boîte à outils en titane","Boîte à outils en élémentium"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Bourse d'herbes","Sac fourre-tout d'herboriste « T vert » par La Tannée","Sac d'herbes cénarien","Sacoche de Cénarius","Sac de botanique de Mycah","Sac émeraude","Sac d'expédition d'Hyjal"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Sacoche d'étudiant « Xandera » par La Tannée","Sacoche de calligraphie","Sac des poches infinies", "Sac de calligraphie bruni"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Pochette brodée de gemmes en exclusivité pour La Tannée","Bourse de gemmes","Sac de joyaux"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Sac en cuir « Meeya » par La Tannée","Sacoche de travailleur du cuir","Sac des nombreuses peaux","Sac de voyage de trappeur", "Sac en cuir bruni"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Sac à métaux précieux « Christina » par La Tannée","Sac de mineur","Sac de mineur renforcé","Sac de mineur colossal", "Sac de mineur bruni"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Boîte d'appâts du maître des leurres"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Réfrigérateur portable", "Unité réfrigérante améliorée"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "Horloge";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "Décalage horaire : ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "Heure locale : ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "Heure serveur : ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "Heure serveur ajustée : ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "Info : clic-gauche pour ajuster le décalage"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "horaire et le format d'heure 12/24H.";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Shift-clic-gauche pour afficher le calendrier.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "Décalage horaire : ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "Décalage";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "  24h";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "Changer le format de l'heure : 12/24 heures";
+L["TITAN_CLOCK_MENU_TEXT"] = "Horloge";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "Afficher l'heure locale (L)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "Afficher l'heure serveur (S)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "Afficher l'heure serveur ajustée (A)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "Afficher tout à droite";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "Cacher le bouton du calendrier";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Cacher le bouton d'horloge";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Cacher le bouton du calendrier";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "Format des coordonnées";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "Pos. : ";
+L["TITAN_LOCATION_TOOLTIP"] = "Info de position";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "Astuce : shift-clic-gauche pour ajouter";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "l'info de position dans les dialogues.";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "Zone : ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "Sous Zone : ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "Info JCJ : ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "Position du foyer";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "Auberge : ";
+L["TITAN_LOCATION_MENU_TEXT"] = "Position";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "Montrer le texte de la zone";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "Montrer les coordonnées sur la carte du monde";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "Curseur : %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "Joueur : %s";
+L["TITAN_LOCATION_NO_COORDS"] = "Pas de coords";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "Montrer la zone au dessus de la minicarte";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Mettre à jour la carte du monde lors de changements de zone";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "IPS : ";
+L["TITAN_FPS_MENU_TEXT"] = "IPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "IPS actuel : ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "IPS moyen : ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "IPS minimum : ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "IPS maximum : ";
+L["TITAN_FPS_TOOLTIP"] = "Images par seconde";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."Ko/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "Latence : ";
+L["TITAN_LATENCY_TOOLTIP"] = "Etat du réseau";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Latence du royaume (domicile) : ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Latence du jeu (monde) : ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "Bande passante sortante : ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "Bande passante entrante : ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Latence";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "Butin : ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "Accès libre";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "Chacun son tour";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "Responsable du butin";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "Butin de groupe";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "Le Besoin avant la Cupidité";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personnel";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "Type de butin";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "Type de butin";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "Lancer les dés";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "Astuce : clic-gauche pour un jet de dés.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "Sélectionnez la plage du lancer de dés via le clic droit.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "Difficulté du donjon";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "Difficulté du raid";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "Afficher la difficulté du donjon/raid";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "Changer la difficulté du donjon";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "Changer la difficulté du raid";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "Auto (basé sur le groupe)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."Mo";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."Ko";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."Ko/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "Mémoire : ";
+L["TITAN_MEMORY_TOOLTIP"] = "Utilisation mémoire";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "Actuel : ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "Initial : ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "Taux d'augmentation : ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "Ko/Mo";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "Performances";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "Performances";
+L["TITAN_PERFORMANCE_ADDONS"] = "Utilisation des addons";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "Utilisation mémoire des addons";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "Format de la mémoire des addons";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "Utilisation CPU des addons";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "Nom:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "Utilisation";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "Taux";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "Mémoire totale des addons :";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "Temps total CPU :";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "Montrer les IPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "Montrer la latence du royaume (domicile)";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Montrer la latence du jeu (monde)";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "Montrer la mémoire";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "Montrer l'utilisation mémoire/CPU des addons";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "Afficher le taux d'occupation mémoire/CPU";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "Analyse de performances CPU ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "Activer l'analyse CPU ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "Désactiver l'analyse CPU ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "Addons surveillés : ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "Addons à surveiller";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "Info : clic-gauche pour forcer un nettoyage de la mémoire.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "XP/h pour ce niveau : ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "XP/h pour cette session : ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "Temps avant niveau : ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "Progression : ";
+L["TITAN_XP_TOTAL_RESTED"] = "Reposé : ";
+L["TITAN_XP_XPTOLEVELUP"] = "XP nécessaire : ";
+L["TITAN_XP_TOOLTIP"] = "Info XP";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "Temps de jeu total : ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "Temps joué pour ce niveau : ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "Temps joué pour cette session : ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "XP totale requise pour ce niveau : ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "XP gagnée pour ce niveau : ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "XP nécessaire pour ce niveau : ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "XP gagnée pour cette session : ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "XP/h pour ce niveau : ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "XP/h pour cette session : ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "Temps avant niveau (taux niveau) : ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "Temps avant niveau (taux session) : ";
+L["TITAN_XP_MENU_TEXT"] = "XP";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "Montrer XP/h pour ce niveau";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "Montrer XP/h pour cette session";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Montrer la vue multi-infos";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "Afficher ...";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "XP reposée";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "XP nécessaire pour ce niveau";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "l'estimation du nombre de mobs à tuer";
+L["TITAN_XP_MENU_RESET_SESSION"] = "Réinitialise la session";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "Rafraîchir les compteurs";
+L["TITAN_XP_UPDATE_PENDING"] = "Mise à jour...";
+L["TITAN_XP_KILLS_LABEL"] = "Mobs à tuer avant niveau (à %s XP par mob) : ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "Mobs à tuer : ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "Temps joué pour cette session : ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "Montrer le temps joué pour cette session";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*) meurt, vous gagnez (%d+) points d'expérience.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "Est. des gains : ";
+L["TITAN_XP_XPGAINS_LABEL"] = "Gains d'XP pour passer de niveau (à %s XP gagné): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "les estimations de gains d'XP pour monter";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "Réparation",
+	tooltip = "Infos de réparation",
+	button = "Durabilité: ",
+	normal = "Coût de réparation (normal) : ",
+	friendly = "Coût de réparation (amical) : ",
+	honored = "Coût de réparation (honoré) : ",
+	revered = "Coût de réparation (révéré) : ",
+	exalted = "Coût de réparation (exalté) : ",
+	buttonNormal = "Afficher normal",
+	buttonFriendly = "Afficher amical (5%)",
+	buttonHonored = "Afficher honoré (10%)",
+	buttonRevered = "Afficher révéré (15%)",
+	buttonExalted = "Afficher exalté (20%)",
+	percentage = "Afficher en pourcentage",
+	itemnames = "Afficher le nom des objets",
+	mostdamaged = "Afficher le plus endommagé",
+	showdurabilityframe = "Afficher la silhouette de durabilité",
+	undamaged = "Afficher les objets intacts",
+	discount = "Afficher la réduction sur le bouton de la barre et les noms d'objets",
+	nothing = "Aucun objet endommagé",
+	confirmation = "Voulez vous réparer tous vos objets ?",
+	badmerchant = "Ce marchand ne peut pas réparer. Affichage des coûts de réparation normaux.",
+	popup = "Afficher la demande de réparation",
+	showinventory = "Réparer aussi l'inventaire",
+	WholeScanInProgress = "Mise à jour...",
+	AutoReplabel = "Réparation Auto",
+	AutoRepitemlabel = "Réparation automatique de tous les objets",
+	ShowRepairCost = "Afficher le coût de réparation",
+	ignoreThrown = "Ignorer les armes de jet",
+	ShowItems = "Afficher les objets",
+	ShowDiscounts = "Afficher les réductions",
+	ShowCosts = "Afficher les coûts",
+	Items = "Objets",
+	Discounts = "Réductions",
+	Costs = "Coûts",
+	CostTotal = "Coût total",
+	CostBag = "Coût de l'inventaire",
+	CostEquip = "Coût de l'équipement",
+	TooltipOptions = "Tooltip",
+};
+L["TITAN_REPAIR"] = "Titan Repair"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "Fonds de la banque de guilde :"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "Retrait autorisé :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "Utiliser les fonds de la banque de guilde"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "La banque de guilde ne peut pas payer les réparations, ou vous ne pouvez pas retirer autant."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "Vous n'êtes soit pas dans une guilde, soit vous n'avez pas le droit d'utiliser la banque de guilde pour réparer vos objets."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "Vous n'avez pas assez pour payer les réparations."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "Annoncer les coût de réparation sur le chat"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "Coût des réparations : "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "Or total sur";
+L["TITAN_GOLD_ITEMNAME"] = "Titan Gold";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "Vider la base de données";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "Remise à zéro de la session";
+L["TITAN_GOLD_DB_CLEARED"] = "Titan Gold - Base vidée.";
+L["TITAN_GOLD_SESSION_RESET"] = "Titan Gold - Session remise à zéro.";
+L["TITAN_GOLD_MENU_TEXT"] = "Gold";
+L["TITAN_GOLD_TOOLTIP"] = "Infos sur les richesses";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "Afficher l'or du joueur";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "Afficher l'or sur le serveur";
+L["TITAN_GOLD_SESS_EARNED"] = "Gagné cette session";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "Gagné par heure";
+L["TITAN_GOLD_SESS_LOST"] = "Perdu cette session";
+L["TITAN_GOLD_PERHOUR_LOST"] = "Perdu par heure";
+L["TITAN_GOLD_STATS_TITLE"] = "Statistiques de la session";
+L["TITAN_GOLD_TTL_GOLD"] = "Or Total";
+L["TITAN_GOLD_START_GOLD"] = "Or de départ";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "Trier par or";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "Trier par nom";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "Afficher l'or gagné par heure";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "Cacher l'or gagné par heure";
+L["TITAN_GOLD_GOLD"] = "o";
+L["TITAN_GOLD_SILVER"] = "a";
+L["TITAN_GOLD_COPPER"] = "c";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "visible";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "caché";
+L["TITAN_GOLD_DELETE_PLAYER"] = "Supprimer ce personnage";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Afficher ce personnage";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "Alliance";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "Horde";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."Attention : "..FONT_COLOR_CODE_CLOSE
+	.."Ceci va réinitialiser toutes les données de Titan Gold. Si vous voulez continuer, cliquez sur 'Accepter', sinon 'Annuler' ou la touche 'Echap'.";
+L["TITAN_GOLD_COIN_NONE"] = "Ne rien afficher";
+L["TITAN_GOLD_COIN_LABELS"] = "Afficher le texte des pièces";
+L["TITAN_GOLD_COIN_ICONS"] = "Afficher les images des pièces";
+L["TITAN_GOLD_ONLY"] = "Afficher uniquement les pièces d'or";
+L["TITAN_GOLD_MERGE"] = "Merge Servers";
+L["TITAN_GOLD_SEPARATE"] = "Separate Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "Contrôle du volume";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "Volume global : ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "Volume des effets : ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "Volume de l'ambiance : ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Volume des discussions : ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "Volume de la musique : ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "Volume du microphone : ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "Volume des voix : ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "Info : clic-gauche pour ajuster"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "le volume sonore.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "Volume : ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "Volume";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "Global";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "Effets";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "Ambiance";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Discussion";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "Musique";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "Micro";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "Voix";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "Haut";
+L["TITAN_VOLUME_CONTROL_LOW"] = "Bas";
+L["TITAN_VOLUME_MENU_TEXT"] = "Ajustement du volume";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "Ouvrir les options Sons et Voix" ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "Remplacer les réglages Blizzard";
diff --git a/Titan/locale/Localization.IT.lua b/Titan/locale/Localization.IT.lua
new file mode 100644
index 0000000..3714e50
--- /dev/null
+++ b/Titan/locale/Localization.IT.lua
@@ -0,0 +1,554 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","itIT")
+if not L then return end
+
+L["TITAN_PANEL"] = "Titan Panel";
+local TITAN_PANEL = "Titan Panel";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/A";
+L["TITAN_SECONDS"] = "secondi";
+L["TITAN_MINUTES"] = "minuti";
+L["TITAN_HOURS"] = "ore";
+L["TITAN_DAYS"] = "giorni";
+L["TITAN_SECONDS_ABBR"] = "s";
+L["TITAN_MINUTES_ABBR"] = "m";
+L["TITAN_HOURS_ABBR"] = "h";
+L["TITAN_DAYS_ABBR"] = "d";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "KB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kbps"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "Nessuno";
+L["TITAN_USE_COMMA"] = "Usa virgola";
+L["TITAN_USE_PERIOD"] = "Usa punto";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "Non puoi eliminare il profilo in uso.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."Attenzione : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "Se vuoi continuare questa operazione premi 'Accetta' (la IU si ricaricherà); altrimenti premi 'Annulla' o il tasto Cancella."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."Questa impostazione reimposterà la tue barre e le impostazioni di "..TITAN_PANEL.." e ricreerà il tuo profilo corrente. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."Questo ricaricherà "..TITAN_PANEL..". "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = "Plugin registrati di "..TITAN_PANEL
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Plugin Registrati"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "I plugin seguenti hanno richiesto di essere registrati su "..TITAN_PANEL..".\n"
+	.."Invia qualsiasi segnalazione di problemi ai rispettivi autori dei plugin."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Tipo"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Categoria"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Nome Pulsante"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Stato"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Problemi"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Note"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Indice tabella"
+L["TITAN_PANEL_EXTRAS"] = TITAN_PANEL.." Extra"
+L["TITAN_PANEL_EXTRAS_SHORT"] = "Extras"
+L["TITAN_PANEL_EXTRAS_DESC"] = "Questi plugin hanno una configurazione ma non sono attivi in questo momento.\n"
+	.."Possono essere eliminati tranquillamente."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "Elimina configurazione"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "la configuazione è stata rimossa."
+L["TITAN_PANEL_CHARS"] = "Personaggi"
+L["TITAN_PANEL_CHARS_DESC"] = "Questi sono personaggi che hanno una configurazione."
+L["TITAN_PANEL_REGISTER_START"] = "Registrazione plugin di "..TITAN_PANEL.." in corso..."
+L["TITAN_PANEL_REGISTER_END"] = "Registrazione plugins completata."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffffReimposta "..TITAN_PANEL.." ai valori/alle posizioni standard.";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffReimposta "..TITAN_PANEL.." la scala dei font dei tooltip.";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffReimposta "..TITAN_PANEL.." la trasparenza dei font dei tooltip.";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffReimposta "..TITAN_PANEL.." la scala.";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffReimposta "..TITAN_PANEL.." la spaziatura tra i pulsanti.";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffffApre la finestra di controllo di "..TITAN_PANEL;
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffffApre the finestra di controllo della trasparenza.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffffApre the finestra di controllo degli skin.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <name> <server>: |cffffffffImposta il profilo selezionando il profilo salvato richiesto.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<name>: |cffffffffpuò essere il nome del personaggio o il nome personalizzato scelto."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<server>: |cffffffffpuò essere il nome del server o 'TitanCustomProfile'."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffToggles "..TITAN_PANEL.." to load silently.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Uso: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." la scala del font dei tooltip è stata reimpostata.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." la trasparenza dei tooltip è stata reimpostata.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." la scala è stata reimpostata.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." la spaziatura tra i bottoni è stata riempostata.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Globale";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Profilo Globale";
+L["TITAN_PANEL_GLOBAL_USE"] = "Usa profilo globale";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Usa come profilo globale";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Usa un profilo che vale per tutti i personaggi";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = " opzioni di reset";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "Non puoi caricare un profilo se stai usando un profilo globale";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffffd700 by the |cffff8c00"..TITAN_PANEL.." Development Team";
+L["TITAN_PANEL_MENU_TITLE"] = TITAN_PANEL;
+L["TITAN_PANEL_MENU_HIDE"] = "Nascondi";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(In combattimento)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(Ricarica l'IU)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "Mostra testo colorato";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "Mostra icona";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "Mostra testo";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "Auto-Nascondi";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "Testo Centrale";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Mostra barra";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "Disabilita adattamento allo schermo";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "Disabilita adattamento alla minimappa";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "Adattamento automatico log";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "Adattamento automatico borse";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "Adattamento automatico alla finestra segnalazioni";
+L["TITAN_PANEL_MENU_PROFILES"] = "Profili";
+L["TITAN_PANEL_MENU_PROFILE"] = "Profilo ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "Personalizzato";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " è stato eliminato.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "Server";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "Personaggio";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "L'IU verrà ricaricata premento 'OK' per permettere il salvataggio del tuo profilo personalizzato.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "Inserisci un nome per il tuo profilo personalizzato:\n(20 caratteri massimo, spazi non permessii, case sensitive)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "Le impostazioni in uso verranno salvate con il nome profilo: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "Il nome profilo inserito esiste già. Sei sicuro di volerlo sovrascrivere? Premi 'Accetta' se si, altrimenti premi 'Annulla' o il tasto 'Cancella.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "Gestisci";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "Carica";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "Elimina";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "Salva";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "Configurazione";
+L["TITAN_PANEL_OPTIONS"] = "Opzioni";
+L["TITAN_PANEL_MENU_TOP"] = "In alto"
+L["TITAN_PANEL_MENU_TOP2"] = "In alto 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "In basso"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "In basso 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = "Tooltip e Finestre di "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Tooltip e Finestre";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Barre in alto"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Barre in basso"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "Barre"
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = "Barre in alto di "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = "Barre in basso di "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Tooltip";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "Finestre";
+L["TITAN_PANEL_MENU_PLUGINS"] = "Plugins";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "Blocca Bottoni";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "Visualizza le versioni dei plugin";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "Plugin di destra";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "Forza il posizionamento dei lanciatori LDB a destra";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"Integrati","Generale","Combattimento","Informazione","Interfaccia","Professioni"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "Mostra tooltip";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "Nascondi tooltip in combattimento";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Blocca le barre che si auto-nascondono in combattimento";
+L["TITAN_PANEL_MENU_RESET"] = "Reimposta "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "Skin";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "Font Finestre"
+L["TITAN_PANEL_MENU_ENABLED"] = "Abilita";
+L["TITAN_PANEL_MENU_DISABLED"] = "Disabilita";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Muovi a Sinistra";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Muovi a Destra";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Mostra testo plugin";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Sempre accesa";
+L["TITAN_PANEL_MENU_POSITION"] = "Posizione";
+L["TITAN_PANEL_MENU_BAR"] = "Barra";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Scegli in quale barra posizionare il plugin";
+L["TITAN_PANEL_MENU_SHOW"] = "Mostra Plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Ricarica Plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Ricarica testo e posizione plugin";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Silent Load";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Versione";
+L["TITAN_ABOUT_AUTHOR"] = "Autore";
+L["TITAN_ABOUT_CREDITS"] = "Crediti";
+L["TITAN_ABOUT_CATEGORY"] = "Categoria";
+L["TITAN_ABOUT_EMAIL"] = "Email";
+L["TITAN_ABOUT_WEB"] = "Sito Web";
+L["TITAN_ABOUT_LICENSE"] = "Licenza";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "Addon che visualizza delle barre di informazioni. Permette agli utenti di aggiungere dei plugin che vengono mostrati in barre nella parte alta e/o bassa dello schermo.";
+L["TITAN_TRANS_MENU_TEXT"] = "Trasparenza di "..TITAN_PANEL;
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "Trasparenza";
+L["TITAN_TRANS_MENU_DESC"] = "Modifica trasparenza per le barre e i tooltip di "..TITAN_PANEL;
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "Barra principale";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "Barra secondaria";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "Tooltip";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "Imposta la trasparenza dei tooltip dei vari plugin.";
+L["TITAN_UISCALE_MENU_TEXT"] = "Scala e Font di "..TITAN_PANEL;
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Scala e Font";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "Scala IU";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "Scala di "..TITAN_PANEL;
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "Spaziatura bottoni";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Spaziatura icone";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "Scala del font dei tooltip";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "Disabilita la scala del font dei tooltip";
+L["TITAN_UISCALE_MENU_DESC"] = "Controlla diversi aspetti dell'IU e di "..TITAN_PANEL..".";
+L["TITAN_UISCALE_SLIDER_DESC"] = "Imposta la scala dell'intera IU.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "Imposta la scala dei vari bottoni e icone di "..TITAN_PANEL..".";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "Modifica lo spazio tra i plugin di sinistra.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Modifica lo spazio tra i plugin di sinistra.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "Modifica la scala dei tooltip dei vari plugin.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "Disabilita il controllo della scala del font dei tooltip di "..TITAN_PANEL..".";
+
+L["TITAN_SKINS_TITLE"] = "Skin di "..TITAN_PANEL;
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Skin Personalizzate";
+L["TITAN_SKINS_TITLE_CUSTOM"] = "Skin personalizzate di "..TITAN_PANEL;
+L["TITAN_SKINS_MAIN_DESC"] = "Tutti gli skin personalizzati devono trovarsi in: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Cartella Skin>\\ ".."\n"
+			.."\nGli skin di "..TITAN_PANEL.." e quelli personalizzati si trovano nella cartella 'Custom'"
+L["TITAN_SKINS_LIST_TITLE"] = "Lista Skin";
+L["TITAN_SKINS_SET_DESC"] = "Scegli uno skin per le barre di "..TITAN_PANEL;
+L["TITAN_SKINS_SET_HEADER"] = "Imposta lo skin di "..TITAN_PANEL;
+L["TITAN_SKINS_RESET_HEADER"] = "Reimposta gli skin di "..TITAN_PANEL;
+L["TITAN_SKINS_NEW_HEADER"] = "Aggiungi un nuovo skin";
+L["TITAN_SKINS_NAME_TITLE"] = "Nome skin"
+L["TITAN_SKINS_NAME_DESC"] = "Inserisci un nome per il nuovo skin. Verrà utilizzato nella lista degli skin.";
+L["TITAN_SKINS_PATH_TITLE"] = "<Cartella Skin>"
+L["TITAN_SKINS_PATH_DESC"] = "<Skin Folder> under the "..TITAN_PANEL.." install. See the example above."
+L["TITAN_SKINS_ADD_HEADER"] = "Aggiungi skin";
+L["TITAN_SKINS_ADD_DESC"] = "Aggiunge un nuovo skin alla lista degli skin disponibili per "..TITAN_PANEL..".";
+L["TITAN_SKINS_REMOVE_HEADER"] = "Rimuovi Skin";
+L["TITAN_SKINS_REMOVE_DESC"] = "Scegli uno skin personalizzato da rimuovere."
+L["TITAN_SKINS_REMOVE_BUTTON"] = "Rimuovi";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "Rimuove lo skin personalizzato scelto.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "Sei responsabile di rimuovere gli skin che non vuoi dalla cartella di installazione di "..TITAN_PANEL..". Agli addon non è permesso rimuovere file."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "Reimposta ai valori standard";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "Reimposta la lista degli skin ai valori standard degli skin di "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "Scegli il tipo di font per i plugin delle barre di "..TITAN_PANEL..".";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "Dimensione Font";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "Imposta la dimensione dei font di "..TITAN_PANEL;
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = ""..TITAN_PANEL.." Frame Strata";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "Adjusts the frame strata for the "..TITAN_PANEL.." Bar(s).";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Avanzate";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Modifica i timer solo se hai problemi con finestre che non si posizionano correttamente.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Ingresso nel mondo";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Modifica valore (di solito aumenta) se le finestre non si posizionano quando entri / esci dal mondo o da un istanza.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Veicolo";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Modifica valore (di solito aumenta) se le finestre non si posizionano quando entri / esci in un veicolo.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "Attiva/Disattiva la funzione Auto-Nascondi di "..TITAN_PANEL;
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "Zaino: ";
+L["TITAN_BAG_TOOLTIP"] = "Info Zaino";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "Suggerimento: Click sinistro per aprire tutti gli zaini.";
+L["TITAN_BAG_MENU_TEXT"] = "Zaino";
+L["TITAN_BAG_USED_SLOTS"] = "Spazi Usati";
+L["TITAN_BAG_FREE_SLOTS"] = "Spazi Liberi";
+L["TITAN_BAG_BACKPACK"] = "Backpack";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "Mostra spazi usati";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "Mostra spazi disponibili";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "Mostra tooltip dettagliato";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "Ignora contenitori";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "Ignora zaini delle professioni";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Borsa di Telamagica Incantata", "Borsa di Telarunica Incantata", "Zaino da Incantatore", "Sacca Grande da Incantatore",
+"Sacca di Fuocoincanto", "Sacca Misteriosa", "Sacca Ultraterrena", "Borsetta da Incantatore \"Esclusiva\" di Dolce Gabbata"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Cassetta degli Attrezzi Pesante", "Cassetta degli Attrezzi di Vilferro", "Cassetta degli Attrezzi di Titanio",
+"Cassetta degli Attrezzi d'Elementio", "Cassetta Tecnologica \"Follia\" di Dolce Gabbata"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Sacca da Erbalista", "Sacca per Erbe Cenariana", "Sacco Cenariano", "Sacca Botanica di Mycah", "Sacca di Smeraldo",
+"Sacca della Spedizione di Hyjal", "Cesta per Erbe \"Verdissima\" di Dolce Gabbata"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Cartella dello Scriba", "Sacca Milletasche", "Zaino da Studente \"Scuola Oggi\" di Dolce Gabbata", "Burnished Inscription Bag"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Borsa da Orafo", "Borsa di Gioielli", "Pochette da Sera \"Inimitabile\" di Dolce Gabbata"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Sacca del Conciatore", "Sacca per Molte Pelli", "Zaino da Viaggio del Mastro Bracconiere", "Sacca di Cuoio \"A Pelle\" di Dolce Gabbata", "Burnished Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Sacco da Minatore", "Sacco da Minatore Rinforzato", "Sacco da Minatore di Mammut", "Sacco di Preziosi \"Diamiani\" di Dolce Gabbata", "Burnished Mining Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Cassetta del Gran Pescatore"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Frigorifero Portatile", "Advanced Refrigeration Unit"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "Orologio";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "Offset ora server: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "Ora locale: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "Ora server: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "Ora server corretta: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "Suggerimento: click sinistro per modificare l'offset dell'ora"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "(solo orario server) e formato ora 12/24.";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Maiusc click sinistro per abilitare/disabilitare il calendario.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "Offset ora server: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "Offset";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24H Fmt";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "Scegle se visualizzare l'ora in formato 12/24 ore";
+L["TITAN_CLOCK_MENU_TEXT"] = "Orologio";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "Mostra ora locale (L)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "Mostra ora server (S)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "Mostra ora server corretta (A)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "Mostra a destra";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "Nascondi bottone ora/calendario";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Nascondi bottone ora";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Nascondi bottone calendario";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "Formato Coordinate";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "Loc: ";
+L["TITAN_LOCATION_TOOLTIP"] = "Info Località";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "Suggerimento: Maiusc + click sinistro per aggiungere informazioni";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "sulla località al messaggio della chat.";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "Zona: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "Sottozona: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "Info PVP: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "Home Location";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "Inn: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "Località";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "Mostra testo zona";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "Mostra coordinate sulla mappa";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "Cursore: %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "Giocatore: %s";
+L["TITAN_LOCATION_NO_COORDS"] = "No Coords";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "Mostra località sopra alla minimappa";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Aggiorna mappa quando cambi località";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "FPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "FPS Correnti: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "FPS Medi: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "FPS Minimi: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "FPS Massimo: ";
+L["TITAN_FPS_TOOLTIP"] = "Frame al secondo";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "Latenza: ";
+L["TITAN_LATENCY_TOOLTIP"] = "Stato Network";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Latenza Reame (locale): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Latenza Gioco (reame): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "Banda in Ingresso: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "Banda in Uscita: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Latenza";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "Predazione: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "Libero per Tutti";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "Rotazione Fissa";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "Resp. del Bottino";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "Bottino di Gruppo";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "Necessità prima di Bramosia";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "Info Tipo Predazione";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "Tipo Predazione";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "Tiro casuale";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "Suggerimento: Click Sinistro per fare un tiro casuale.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "Scegli tipo di tiro dal menu col tasto destro.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "Difficolta Scorreria";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "Difficoltà Incursione";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "Mostra difficoltà Scorreria/Incursione";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "Imposta difficoltà Scorreria";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "Imposta difficoltà Incursione";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "Auto (in base al gruppo)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "Memoria: ";
+L["TITAN_MEMORY_TOOLTIP"] = "Uso Memoria";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "Corrente: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "Iniziale: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "Velocità aumento: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "Info Performance";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "Performance";
+L["TITAN_PERFORMANCE_ADDONS"] = "Uso Addon";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "Uso memoria Addon";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "Formato memoria Addon";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "Uso CPU Addon";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "Nome:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "Uso";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "Tasso";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "Memoria Addon totale:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "Tempo CPU totale:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "Mostra FPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "Mostra latenza reame";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Mostra latenza gioco";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "Mostra memoria";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "Mostra uso memoria addon";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "Mostra tasso uso Addon";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "Modalità controllo CPU";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "Abilita modalità controllo CPU ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "Disabilita modalità controllo CPU ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "Addon monitorati: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "Addon monitorati";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "Suggerimento: Click sinistro per forzare una garbage collection.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "PE/hr questo livello: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "PE/hr questa sessione: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "Tempo per salire di lv: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "Livello completo: ";
+L["TITAN_XP_TOTAL_RESTED"] = "Riposato: ";
+L["TITAN_XP_XPTOLEVELUP"] = "PE al prossimo lv: ";
+L["TITAN_XP_TOOLTIP"] = "Info PE";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "Tempo totale di gioco: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "Tempo a questo lv: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "Tempo questa sessione: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "PE totali necessari questo lv: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "PE guadagnati questo lv: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "PE necessari al prossimo lv: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "PE guadagnati questa sessione: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "PE/HR questo lv: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "PE/HR questa sessione: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "Tempo al prossimo lv (Tasso lv): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "Tempo al prossimo lv (Tasso sessione): ";
+L["TITAN_XP_MENU_TEXT"] = "PE";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "Mostra PE/HR questo lv";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "Mostra PE/HR questa sessione";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Mostra vista multi-info";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "Bottone";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "Mostra PE Riposato";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "Mostra PE al prossimo lv";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "Mostra uccisioni rimanenti al prossimo lv";
+L["TITAN_XP_MENU_RESET_SESSION"] = "Reimposta sessione";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "Reimposta timers";
+L["TITAN_XP_UPDATE_PENDING"] = "Aggiorno...";
+L["TITAN_XP_KILLS_LABEL"] = "Uccisioni al prossimo lv (%s PE ottenuti l'ultima volta): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "Uccisioni: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "Tempo sessione: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "Mostra tempo sessione";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*) muore, otteni (%d+) esperienza.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "Incrementi PE da ott.: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "Incrementi PE al prossimo lv. (%s PE ottenuti l'ultima volta): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "Mostra incrementi PE al prossimo lv";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "Riparazione",
+	tooltip = "Info Riparazione",
+	button = "Integrità: ",
+	normal = "Costo Riparazione (Normale): ",
+	friendly = "Costo Riparazione (Amichevole): ",
+	honored = "Costo Riparazione (Onorato): ",
+	revered = "Costo Riparazione (Riverito): ",
+	exalted = "Costo Riparazione (Osannato): ",
+	buttonNormal = "Mostra Normale",
+	buttonFriendly = "Mostra Amichevole (5%)",
+	buttonHonored = "Mostra Onorato (10%)",
+	buttonRevered = "Mostra Riverito (15%)",
+	buttonExalted = "Mostra Osannato (20%)",
+	percentage = "Mostra come percentuale",
+	itemnames = "Mostra nomi oggetti",
+	mostdamaged = "Mostra più danneggiato",
+	showdurabilityframe = "Mostra integrità",
+	undamaged = "Mostra oggetti non danneggiati",
+	discount = "Sconto",
+	nothing = "Niente da riparare",
+	confirmation = "Vuoi riparare tutti gli oggetti?",
+	badmerchant = "Questo mercante non può riparare. Visualizzo i costi normali di riparazione.",
+	popup = "Mostra popup riparazione",
+	showinventory = "Calcola danno nello zaino",
+	WholeScanInProgress = "Aggiorno...",
+	AutoReplabel = "Auto-Ripara",
+	AutoRepitemlabel = "Ripara automaticamente tutti gli oggetti",
+	ShowRepairCost = "Mostra costo riparazione",
+	ignoreThrown = "Ignora coltelli da lancio",
+	ShowItems = "Mostra oggetti",
+	ShowDiscounts = "Mostra sconti",
+	ShowCosts = "Mostra costi",
+	Items = "Oggetti",
+	Discounts = "Sconti",
+	Costs = "Costi",
+	CostTotal = "Costo totale",
+	CostBag = "Costo zaino",
+	CostEquip = "Costo equipaggiati",
+	TooltipOptions = "Tooltip",
+};
+
+L["TITAN_REPAIR"] = "Riparazione di Titan"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "Fondi della banca della gilda:"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "Fondi disponibili della banca della gilda:"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "Usa fondi della banca della gilda"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "La banca della gilda non può permettersi così tanto oppure non hai i permessi per prendere così tanti soldi."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "Non sei in una gilda o non hai il permesso per utilizzare la banca della gilda per riparare i tuoi oggettix."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "Non puoi permetterti la riparazione."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "Scrivi il costo di riparazione nella chat"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "Il costo di riparazione è "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "Oro totale su";
+L["TITAN_GOLD_ITEMNAME"] = "Titan Gold";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "Svuota Database";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "Reimposta sessione corrente";
+L["TITAN_GOLD_DB_CLEARED"] = "Titan Gold - Database cancellato.";
+L["TITAN_GOLD_SESSION_RESET"] = "Titan Gold - Reimposta sessione.";
+L["TITAN_GOLD_MENU_TEXT"] = "Oro";
+L["TITAN_GOLD_TOOLTIP"] = "Info oro";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "Mostra oro giocatore";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "Mostra oro server";
+L["TITAN_GOLD_SESS_EARNED"] = "Guadagnati in questa sessione";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "Guadagnati ogni ora";
+L["TITAN_GOLD_SESS_LOST"] = "Persi questa sessione";
+L["TITAN_GOLD_PERHOUR_LOST"] = "Persi ogni ora";
+L["TITAN_GOLD_STATS_TITLE"] = "Statistiche sessione";
+L["TITAN_GOLD_TTL_GOLD"] = "Oro totale";
+L["TITAN_GOLD_START_GOLD"] = "Oro iniziale";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "Ordina per quantità";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "Ordina per nome";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "Mostra oro ogni ora";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "Nascondi oro per ora";
+L["TITAN_GOLD_GOLD"] = "g";
+L["TITAN_GOLD_SILVER"] = "s";
+L["TITAN_GOLD_COPPER"] = "c";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "Visibile";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "Nascosto";
+L["TITAN_GOLD_DELETE_PLAYER"] = "Elimina personaggio";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Mostra personaggio";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "Alleanza";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "Orda";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."Attenzione: "
+..FONT_COLOR_CODE_CLOSE.."Questa impostazione cancellerà il database. "
+.."Se vuoi continuare con questa operazione, premi 'Accetta', altrimenti premi 'Annulla' o il tasto Cancella.";
+L["TITAN_GOLD_COIN_NONE"] = "Non mostrare descrizione";
+L["TITAN_GOLD_COIN_LABELS"] = "Mostra descrizione";
+L["TITAN_GOLD_COIN_ICONS"] = "Mostra icone";
+L["TITAN_GOLD_ONLY"] = "Mostra solo oro";
+L["TITAN_GOLD_COLORS"] = "Mostra colori";
+L["TITAN_GOLD_MERGE"] = "Merge Servers";
+L["TITAN_GOLD_SEPARATE"] = "Separate Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "Info Volume";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "Volume master: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "Volume effetti: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "Volume effetti ambiente: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "Volume suono musica: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "Volume microfono: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "Volume casse: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "Suggerimento: Click Sinistro per impostare il"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "volume.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "Controllo volume: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "Controllo volume";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "Master";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "Effetti";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "Ambiente";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "Musica";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "Microfono";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "Casse";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "Alto";
+L["TITAN_VOLUME_CONTROL_LOW"] = "Basso";
+L["TITAN_VOLUME_MENU_TEXT"] = "Controllo volume";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "Mostra opzioni suono/voce" ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "Sovrascrivi impostazioni volume Blizzard";
diff --git a/Titan/locale/Localization.KR.lua b/Titan/locale/Localization.KR.lua
new file mode 100644
index 0000000..d84987e
--- /dev/null
+++ b/Titan/locale/Localization.KR.lua
@@ -0,0 +1,555 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","koKR")
+if not L then return end
+
+L["TITAN_PANEL"] = "타이탄 패널";
+local TITAN_PANEL = "타이탄 패널";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "없음";
+L["TITAN_SECONDS"] = "초";
+L["TITAN_MINUTES"] = "분";
+L["TITAN_HOURS"] = "시";
+L["TITAN_DAYS"] = "일";
+L["TITAN_SECONDS_ABBR"] = "초";
+L["TITAN_MINUTES_ABBR"] = "분";
+L["TITAN_HOURS_ABBR"] = "시";
+L["TITAN_DAYS_ABBR"] = "일";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "KB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kbps"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "없음";
+L["TITAN_USE_COMMA"] = "쉼표 사용";
+L["TITAN_USE_PERIOD"] = "마침표 사용";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "현재 프로필을 삭제할 수 없습니다.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."경고 : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "이명령을 계속하려면 확인 버튼을 누르고(UI 재시작), 그렇지 않으면 취소 또는 ESC키를 누르세요."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."이 설정은 바를 초기화 합니다 또한 바 설정값도 초기화 하여 프로필을 재설정합니다. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."타이탄 패널을 재시작해야 합니다. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = TITAN_PANEL.." 시도"
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "시도"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "아래의 플러그인을 등록하도록 요청합니다. "..TITAN_PANEL..".\n"
+	.."플러그인 제작자에게 오류를 보고 하세요."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "유형"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "항목 분류"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "버튼 이름"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "상태"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Issue"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "노트"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Table index"
+L["TITAN_PANEL_EXTRAS"] = TITAN_PANEL.." 추가"
+L["TITAN_PANEL_EXTRAS"] = "추가"
+L["TITAN_PANEL_EXTRAS_DESC"] = "데이터 설정을 위한 플러그인을 불러올 수 없습니다.\n"
+	.."노트: 리스트의 변경을 위해서 게임종료 해야합니다."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "설정 데이터 삭제"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "삭제를 위한 설정 입력입니다."
+L["TITAN_PANEL_CHARS"] = "캐릭터"
+L["TITAN_PANEL_CHARS_DESC"] = "데이터 설정에 있는 캐릭터입니다.\n"
+L["TITAN_PANEL_REGISTER_START"] = "타이탄 플러그인 등록..."
+L["TITAN_PANEL_REGISTER_END"] = "프로세스 등록 완료"
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."사용법: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffff초기화 - "..TITAN_PANEL.." 기본 값 및 위치";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffff초기화 - "..TITAN_PANEL.."툴팁 글꼴 크기";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffff초기화 - "..TITAN_PANEL.."툴팁 투명도";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffff초기화 - "..TITAN_PANEL.."크기";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffff초기화 - "..TITAN_PANEL.."버튼 간격";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."사용법: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffff열기 - "..TITAN_PANEL.." 패널 설정창";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffff투명도 설정창을 엽니다.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffff스킨 설정창을 엽니다.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."사용법: |cffffffff/titan {사용하려는 프로필 이름 <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."프로필 사용 <name> <server>: |cffffffff저장된 프로필로 현재 프로필을 설정합니다.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<name>: |cffffffff캐릭터 이름 또는 다른 이름으로 프로필을 변경할 수 있습니다."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<server>: |cffffffff현재 서버이름 또는 'TitanCustomProfile'로 프로필을 변경할 수 있습니다."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffToggles "..TITAN_PANEL.." to load silently.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."사용법: |cffffffff/titan {help | 도움말 <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."사용법: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." 툴팁 글꼴 크기가 초기화 되었습니다.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." 툴팁 투명도가 초기화 되었습니다.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." 크기가 초기화 되었습니다.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." 버튼 공간이 초기화되었습니다.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "공통";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "공통 프로필";
+L["TITAN_PANEL_GLOBAL_USE"] = "공통 프로필 사용";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "공통 프로필로 사용합니다.";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "모든 캐릭터를 공통 프로필로 사용합니다.";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "설정 초기화";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "공통 프로필로 사용할때 불러오지 못할 수도 있습니다.";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffff8c00"..TITAN_PANEL.." |cffffd700개발팀";
+L["TITAN_PANEL_MENU_TITLE"] = "타이판 패널";
+L["TITAN_PANEL_MENU_HIDE"] = "숨김";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(전투중)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(UI 재시작)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "색상화 글자 표시";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "아이콘 표시";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "라벨 글자 표시";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "자동 숨김";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "중앙 글자";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "바 표시";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "화면 보정 불가";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "미니맵 보정 불가";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "자동 로그 보정";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "자동 가방 보정";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "자동 프레임 보정";
+L["TITAN_PANEL_MENU_PROFILES"] = "프로필";
+L["TITAN_PANEL_MENU_PROFILE"] = "프로필 ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "사용자";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " 삭제되었습니다.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "서버";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "캐릭터";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "사용자 프로필을 저장하려면 UI를 재시작해야 합니다. 확인 버튼을 누르세요.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "사용자 프로필의 이름을 입력하세요.:\n(한글 10글자, 영문 20글자, 띄어쓰기 없음, 대소문자 구별)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "현재 설정이 저장될 프로필 이름: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "입력한 프로필 이름이 있습니다. 덮어씌우시겠습니까? 덮어씌우려면 확인 버튼을 누루시고, 취소하려면 취소 버튼이나 ESC키를 눌러주세요.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "관리";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "불러옴";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "삭제";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "저장";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "설정";
+L["TITAN_PANEL_OPTIONS"] = "설정";
+L["TITAN_PANEL_MENU_TOP"] = "위"
+L["TITAN_PANEL_MENU_TOP2"] = "위 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "아래"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "아래 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = TITAN_PANEL.." 툴팁과 창";
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "툴팁과 창";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "상단 바들"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "하단 바들"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "바"
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = TITAN_PANEL.." 바";
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = TITAN_PANEL.." Aux 바";
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "툴팁";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "프레임";
+L["TITAN_PANEL_MENU_PLUGINS"] = "플러그인";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "버튼 고정";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "플러그인 버전 표시";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "오른쪽 플러그인";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "LDB 실행을 오른쪽으로 배치 합니다.";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"기본 내장","일반","전투","정보","기능강화","전문기술"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "툴팁 표시";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "전투중 툴팁 숨기기";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "전투시 바를 자동으로 숨깁니다.";
+L["TITAN_PANEL_MENU_RESET"] = "패널을 초기값으로 되돌립니다.";
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "스킨";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "패널 글꼴"
+L["TITAN_PANEL_MENU_ENABLED"] = "사용";
+L["TITAN_PANEL_MENU_DISABLED"] = "중지";
+L["TITAN_PANEL_SHIFT_LEFT"] = "쉬프트 왼쪽";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "쉬프트 오른쪽";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "플러그인 글자 표시";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "항상 켬";
+L["TITAN_PANEL_MENU_POSITION"] = "위치";
+L["TITAN_PANEL_MENU_BAR"] = "바";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "바에 표시";
+L["TITAN_PANEL_MENU_SHOW"] = "플러그인 표시";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "플러그인 갱신";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "플러그인 글자와 위치를 갱신합니다.";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Silent Load";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "버전";
+L["TITAN_ABOUT_AUTHOR"] = "제작자";
+L["TITAN_ABOUT_CREDITS"] = "만든이";
+L["TITAN_ABOUT_CATEGORY"] = "분류";
+L["TITAN_ABOUT_EMAIL"] = "이메일";
+L["TITAN_ABOUT_WEB"] = "홈페이지";
+L["TITAN_ABOUT_LICENSE"] = "저작권";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "타이탄 바에 대한 정보입니다. 유저에게 필요한 정보 또는 기능을 여러 플러그인을 통해 바에 표시하고 변겅할 수 있는 버튼등을 위 또는 아래에 표시합니다.";
+L["TITAN_TRANS_MENU_TEXT"] = TITAN_PANEL.." 투명도";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "투명도";
+L["TITAN_TRANS_MENU_DESC"] = "타이탄 및 툴팁의 투명도를 조절합니다.";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "메인바";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "보조바";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "툴팁";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "여러 플러그인 툴팁의 투명도를 조절합니다.";
+L["TITAN_UISCALE_MENU_TEXT"] = TITAN_PANEL.." 크기와 글꼴";
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "크기와 글꼴";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "UI 크기";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "패널 크기";
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "버튼 간격";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "아이콘 간격";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "툴팁 글꼴 크기";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "툴팁 글꼴 크기 불가";
+L["TITAN_UISCALE_MENU_DESC"] = "UI 및 패널의 여러가지를 조절합니다.";
+L["TITAN_UISCALE_SLIDER_DESC"] = "전체 UI의 크기를 설정합니다.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "패널의 버튼 및 아이콘의 크기를 설정합니다.";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "왼쪽의 플러그인의 공간을 조절합니다.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "오른쪽의 플러그인의 간격을 조절합니다.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "다양한 플러그인의 툴팁의크기를 조절합니다.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "타이탄 툴팁의 글꼴 크기 조절이 불가능 합니다.";
+
+L["TITAN_SKINS_TITLE"] = TITAN_PANEL.." 스킨";
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "스킨 - 선택";
+L["TITAN_SKINS_TITLE_CUSTOM"] = TITAN_PANEL.." 사용자 스킨";
+L["TITAN_SKINS_MAIN_DESC"] = "모든 사용자 스킨은 정아래 경로에 있어야 합니다 : \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ ".."\n"
+			.."\n"..TITAN_PANEL.." 과 함께 사용자 스킨을 불러올 수 있습니다."
+L["TITAN_SKINS_LIST_TITLE"] = "스킨 목록";
+L["TITAN_SKINS_SET_DESC"] = "타이탄의 설정을 위한 스킨을 선택하세요.";
+L["TITAN_SKINS_SET_HEADER"] = "패널 스킨 설정";
+L["TITAN_SKINS_RESET_HEADER"] = "초기화 - "..TITAN_PANEL.." 스킨";
+L["TITAN_SKINS_NEW_HEADER"] = "새로운 스킨 추가";
+L["TITAN_SKINS_NAME_TITLE"] = "스킨 이름";
+L["TITAN_SKINS_NAME_DESC"] = "새로운 스킨에 이름을 입력하세요.";
+L["TITAN_SKINS_PATH_TITLE"] = "스킨 경로";
+L["TITAN_SKINS_PATH_DESC"] = "정확한 스킨 경로를 입력합니다(아래 예제를 참고하세요).";
+L["TITAN_SKINS_ADD_HEADER"] = "스킨 추가";
+L["TITAN_SKINS_ADD_DESC"] = "패널의 가능한 스킨 목록에 새로운 스킨을 추가합니다.";
+L["TITAN_SKINS_REMOVE_HEADER"] = "스킨 삭제";
+L["TITAN_SKINS_REMOVE_DESC"] = "패널의 가능한 스킨에서 삭제할 스킨을 선택합니다.";
+L["TITAN_SKINS_REMOVE_BUTTON"] = "삭제";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "패널의 가능한 스킨 목록에 선택한 스킨을 삭제합니다.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "원하지 않는 스킨을 삭제할 수 있습니다. "
+	..""..TITAN_PANEL.." 설치된 폴더에서 삭제합니다."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "기본값으로 초기화";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "모든 스킨을 기본값으로 되돌립니다.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "타이탄 플러그인의 글꼴을 변경합니다.";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "글꼴 크기";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "패널의 글꼴 크기를 변경합니다.";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = "패널 프레임 레벨";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "타이탄의 프레임 레벨을 조정합니다.";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Advanced";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Change Timers only if you experience issues with frames not adjusting.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "월드 진이붕";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving world or an instance.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "탈 것";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving vehicle.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "패널 자동 숨김을 전환합니다.";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "가방: ";
+L["TITAN_BAG_TOOLTIP"] = "가방 정보";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "힌트: 클릭하면 모든 가방을 엽니다.";
+L["TITAN_BAG_MENU_TEXT"] = "가방";
+L["TITAN_BAG_USED_SLOTS"] = "사용한 공간";
+L["TITAN_BAG_FREE_SLOTS"] = "남은 공간";
+L["TITAN_BAG_BACKPACK"] = "소지품";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "사용한 공간 표시";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "남은 공간 표시";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "상세 툴팁 표시";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "특수 가방 제외";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "전문기술 가방 제외";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"마력 깃든 마법매듭 가방", "마력 깃든 룬매듭 가방", "마법부여사의 가방", "큰 마법용품 가방", "마법불꽃 가방",
+"불가사의한 가방", "다른 세상의 가방", "\"나르지 독점 상품\" 마법부여 나들이 지갑"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"무거운 공구함", "지옥무쇠 연장통", "티타늄 연장통", "엘레멘티움 연장통", "\"나르지 아가씨용\" 최첨단 가방"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"약초 주머니", "세나리온 약초 가방", "세나리우스의 주머니", "미카의 채집용 주머니", "에메랄드 가방", "하이잘 원정대 가방",
+"\"나르지 자연주의\" 약초용 손가방"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"주문각인사의 가방", "무한 주머니 가방", "\"나르지 실용주의\" 학생 가방", "Burnished Inscription Bag"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"보석 주머니", "보석 자루", "\"나르지 독점 상품\" 보석 박힌 손지갑"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"가죽세공인의 가방", "가죽 덧댄 가방", "덫사냥꾼의 여행 가방", "\"나르지 보들보들\" 가죽 가방", "Burnished Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"강화된 채광용 자루", "채광용 자루", "매머드 채광 자루", "\"나르지 찰캉찰캉\" 귀중한 금속 가방", "Burnished Mining Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"미끼 낚시의 대가 낚시상자"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Portable Refrigerator", "Advanced Refrigeration Unit"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "시계";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "서버 기준 시간 값: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "지역 시간: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "서버 시간: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "서버 시간 보정: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "힌트: 왼쪽 클릭으로 시간을 보정합니다."
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "(서버 시간) 12/24시간 형식으로 변경";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "쉬프트 클릭으로 달력을 엽니다.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "서버 시간 기준: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "기준";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24H Fmt";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "시간 표시를 12/24시간 형식으로 변경합니다.";
+L["TITAN_CLOCK_MENU_TEXT"] = "시계";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "지역 시간 표시";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "서버 시간 표시";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "서버 시간 보정을 표시";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "오른쪽에 표시";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "시간/달력 버튼 숨기기";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "시간 버튼 숨기기";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "달력 버튼 숨기기";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "좌표 형식";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "위치: ";
+L["TITAN_LOCATION_TOOLTIP"] = "지역 정보";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "힌트: 대화창에 현재 위치를";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "쉬프트 + 클릭으로 입력합니다.";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "지역: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "세부지역: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "PVP 정보: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "여관 위치";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "여관: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "좌표";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "지역 표시";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "세계 지도에 좌표 표시";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "커서(X,Y): %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "플레이어(X,Y): %s";
+L["TITAN_LOCATION_NO_COORDS"] = "좌표 없음";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "미니맵에 위치를 표시합니다.";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "FPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "현재 FPS: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "평균 FPS: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "최소 FPS: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "최대 FPS: ";
+L["TITAN_FPS_TOOLTIP"] = "시간당 프레임율";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "지연시간: ";
+L["TITAN_LATENCY_TOOLTIP"] = "네트워크 상태";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "서버 지연시간(집): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "게임 지연시간(세계): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "입력 대역폭: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "출력 대역폭: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "지연시간";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "루팅: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "자유 획득";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "차례대로 획득";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "담당자 획득";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "파티 획득";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "획득 전에 묻기";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "전리품 획득 정보";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "전리품 획득 방식";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "랜덤 주사위";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "힌트: 왼쪽 클릭으로 랜덤 주사위를 굴립니다.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "오른쪽 클릭으로 획득 방법을 선택합니다.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "던전 난이도";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "공격대 난이도";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "공격대/던전 난이도 표시";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "던전 난이도 설정";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "공격대 난이도 설정";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "자동 (파티 기본)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "메모리: ";
+L["TITAN_MEMORY_TOOLTIP"] = "메모리 사용량";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "현재: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "초기: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "증가율: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "성능 정보";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "성능";
+L["TITAN_PERFORMANCE_ADDONS"] = "애드온 사용";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "애드온 메모리 사용량";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "애드온 메모리 포맷";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "애드온 CPU 사용량";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "이름:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "사용";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "비율";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "총 애드온 메모리:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "총 CPU 시간:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "FPS 표시";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "서버 지연시간 표시";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "게임 지연시간 표시";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "메모리 표시";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "애드온 메모리 사용량 표시";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "애드온 사용 비율 표시";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "CPU 프로필 모드";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "CPU 프로필 모드 시작 ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "CPU 프로필 모드 중지 ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "표시할 애드온: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "애드온 갯수";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "힌트: 왼쪽 클릭으로 메모리를 청소합니다.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "현재 레벨에서의 시간당 경험치: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "현재 접속에서의 시간당 경험치: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "레벨당 시간: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "레벨 완료: ";
+L["TITAN_XP_TOTAL_RESTED"] = "휴식경험치: ";
+L["TITAN_XP_XPTOLEVELUP"] = "레벨업까지 경험치: ";
+L["TITAN_XP_TOOLTIP"] = "경험치 정보";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "총 플레이 시간: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "현재 레벨에서의 플레이 시간: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "현재 접속에서의 플레이 시간: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "현재 레벨에서 필요한 총 경험치: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "현재 레벨에서 획득한 경험치: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "다음 레벨까지 필요한 경험치: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "현재 접속에서 획득한 경험치: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "현재 레벨에서 시간당 경험치: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "현재 접속에서 시간당 경험치: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "레벨당 시간 (레벨 비율): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "레벨당 시간 (접속 비율): ";
+L["TITAN_XP_MENU_TEXT"] = "경험치";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "현재 레벨에서의 시간당 경험치 표시";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "현재 접속에서의 시간당 경험치 표시";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Show Multi-Info View";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "버튼";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "휴식경험치 표시";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "다음 레벨까지 경험치 표시";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "다음 레벨까지 필요한 킬수 표시";
+L["TITAN_XP_MENU_RESET_SESSION"] = "현재 접속 초기화";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "갱신 타이머";
+L["TITAN_XP_UPDATE_PENDING"] = "갱신중...";
+L["TITAN_XP_KILLS_LABEL"] = "다음 레벨까지 필요한 킬수 (최근 획득한 경험치 %s 기준): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "킬 수: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "접속 시간: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "현재 접속 시간 표시";
+L["TITAN_XP_GAIN_PATTERN"] = "%1$s|1이;가; 죽었습니다. %2$d의 경험치를 획득했습니다.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "획득: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "레벨당 획득한 경험치 (최근 획득한 경험치 %s 기준): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "레벨당 경험치 획득 표시";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "수리",
+	tooltip = "내구도 정보",
+	button = "내구도: ",
+	normal = "수리비용 (중립적): ",
+	friendly = "수리비용 (약간 우호적): ",
+	honored = "수리비용 (우호적): ",
+	revered = "수리비용 (매우 우호적): ",
+	exalted = "수리비용 (확고한 동맹): ",
+	buttonNormal = "중립적 표시",
+	buttonFriendly = "약간 우호적 표시 (5%)",
+	buttonHonored = "우호적 표시 (10%)",
+	buttonRevered = "매우 우호적 표시 (15%)",
+	buttonExalted = "확고한 동맹 표시 (20%)",
+	percentage = "백분율로 표시",
+	itemnames = "아이템 이름 표시",
+	mostdamaged = "최대 공격력 표시",
+	showdurabilityframe = "내구도 프레임 표시",
+	undamaged = "내구도 100% 아이템 표시",
+	discount = "할인",
+	nothing = "내구도 없음",
+	confirmation = "모든 아이템을 수리하시겠습니까 ?",
+	badmerchant = "이 상인은 수리가 가능하지 않습니다. 대신에 수리 비용을 표시합니다.",
+	popup = "수리 팝업 창 표시",
+	showinventory = "가방 내구도 조사",
+	WholeScanInProgress = "갱신중...",
+	AutoReplabel = "자동 수리",
+	AutoRepitemlabel = "모든 아이템을 자동 수리 합니다.",
+	ShowRepairCost = "수리비용 표시",
+	ignoreThrown = "투척 무기 제외",
+	ShowItems = "아이템 표시",
+	ShowDiscounts = "할인 표시",
+	ShowCosts = "수리비용 표시",
+	Items = "아이템",
+	Discounts = "할인",
+	Costs = "수리비용",
+	CostTotal = "총 비용",
+	CostBag = "가방 비용",
+	CostEquip = "장착 비용",
+	TooltipOptions = "툴팁",
+};
+
+L["TITAN_REPAIR"] = "타이탄 수리"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "길드 은행 금액 :"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "길드 은행 취소 수락 :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "길드 은행 사용"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "길드 은행의 잔액이 부족하여 수리가 불가능합니다."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "길드가 없거나 길드 은행의 비용으로 수리할 권한이 없습니다."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "길드 은행의 잔액이 부족하여 현재는 수리할 수 없습니다."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "대화창에 수리비용 표시"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "수리비용: "
+
+L["TITAN_PLUGINS_MENU_TITLE"] = "플러그인";
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "타이탄 소지금 켬";
+L["TITAN_GOLD_ITEMNAME"] = "소지금";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "데이터베이스 초기화";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "현재 접속 초기화";
+L["TITAN_GOLD_DB_CLEARED"] = "타이탄 소지금 - 데이터베이스 초기화.";
+L["TITAN_GOLD_SESSION_RESET"] = "타이탄 소지금 - 현재 접속 초기화.";
+L["TITAN_GOLD_MENU_TEXT"] = "소지금";
+L["TITAN_GOLD_TOOLTIP"] = "소지금 정보";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "플레이어 소지금 표시";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "계정 소지금 표시";
+L["TITAN_GOLD_SESS_EARNED"] = "현재 접속 획득";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "시간당 획득";
+L["TITAN_GOLD_SESS_LOST"] = "현재 접속 사용";
+L["TITAN_GOLD_PERHOUR_LOST"] = "시간당 사용";
+L["TITAN_GOLD_STATS_TITLE"] = "접속 정보";
+L["TITAN_GOLD_TTL_GOLD"] = "타이탄 소지금";
+L["TITAN_GOLD_START_GOLD"] = "소지금 시작";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "소지금에 따라 정렬";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "이름에 따라 정렬";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "시간당 획득 표시";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "시간당 획득 숨김";
+L["TITAN_GOLD_GOLD"] = "G";
+L["TITAN_GOLD_SILVER"] = "S";
+L["TITAN_GOLD_COPPER"] = "C";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "표시";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "숨김";
+L["TITAN_GOLD_DELETE_PLAYER"] = "캐릭터 삭제";
+L["TITAN_GOLD_SHOW_PLAYER"] = "캐릭터 표시";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "얼라이언스";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "호드";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."경고: "
+..FONT_COLOR_CODE_CLOSE.."이 설정은 소지금 정보를 모두 삭제합니다."
+.." 삭제하려면 확인을 취소하려면 취소 또는 ESC 키를 누르세요.";
+L["TITAN_GOLD_COIN_NONE"] = "라벨 없음 표시";
+L["TITAN_GOLD_COIN_LABELS"] = "동전 이름 표시";
+L["TITAN_GOLD_COIN_ICONS"] = "아이콘 라벨 표시";
+L["TITAN_GOLD_ONLY"] = "소지금만 표시";
+L["TITAN_GOLD_COLORS"] = "소지금 색상 표시";
+L["TITAN_GOLD_MERGE"] = "서버 통합";
+L["TITAN_GOLD_SEPARATE"] = "서버 분리";
+
+L["TITAN_VOLUME_TOOLTIP"] = "음량 정보";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "주 음량 : ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "효과음: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "환경 소리: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "배경음악: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "마이크: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "스피커: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "힌트: 왼쪽 클릭하면 "
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "음량 조절 메뉴를 엽니다.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "음량 조절: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "음량 조절";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "주 음량";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "효과음";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "환경 소리";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "배경음악";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "마이크";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "스피커";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "큼";
+L["TITAN_VOLUME_CONTROL_LOW"] = "작음";
+L["TITAN_VOLUME_MENU_TEXT"] = "음량 조절";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "음량/효과 설정을 표시합니다." ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "블리자드 소리 설정을 덮어 씌웁니다.";
diff --git a/Titan/locale/Localization.RU.lua b/Titan/locale/Localization.RU.lua
new file mode 100644
index 0000000..193b86a
--- /dev/null
+++ b/Titan/locale/Localization.RU.lua
@@ -0,0 +1,554 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","ruRU")
+if not L then return end
+
+L["TITAN_PANEL"] = "Титан Панель";
+local TITAN_PANEL = "Титан Панель";
+L["TITAN_DEBUG"] = "<Титан>";
+L["TITAN_PRINT"] = "Титан";
+
+L["TITAN_NA"] = "N/A";
+L["TITAN_SECONDS"] = "секунды";
+L["TITAN_MINUTES"] = "минуты";
+L["TITAN_HOURS"] = "часы";
+L["TITAN_DAYS"] = "дни";
+L["TITAN_SECONDS_ABBR"] = "с";
+L["TITAN_MINUTES_ABBR"] = "м";
+L["TITAN_HOURS_ABBR"] = "ч";
+L["TITAN_DAYS_ABBR"] = "д";
+L["TITAN_MILLISECOND"] = "мс";
+L["TITAN_KILOBYTES_PER_SECOND"] = "КБ/с";
+L["TITAN_KILOBITS_PER_SECOND"] = "кбит/с"
+L["TITAN_MEGABYTE"] = "мб";
+L["TITAN_NONE"] = "Нету";
+L["TITAN_USE_COMMA"] = "Use comma";
+L["TITAN_USE_PERIOD"] = "Use period";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "Вы не можете удалить свой текущий профиль.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."Внимание : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "Если вы желаете продолжить данную операцию, нажмите 'Accept' (ваш UI будет перезагружен), емли нет, жмите 'Cancel' или клавишу 'Escape'."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."Данные настройки сбросять ваши панели на стандартные настройки и обновят ваш текущий профиль. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."Это перезагрузит Titan Panel. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = "Подчинения"
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Подчинения"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "Ниже, приведены плагины, которые зарегистрировались в Titanе.\n"
+	.."Пожалуйста, любые вопросы отсылайте автору плагина."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Тип"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Категория"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Название кнопки"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Статус"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Проблема"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Заметки"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Индекс"
+L["TITAN_PANEL_EXTRAS"] = "Дополнения"
+L["TITAN_PANEL_EXTRAS_SHORT"] = "Дополнения"
+L["TITAN_PANEL_EXTRAS_DESC"] = "Есть плагины с данными конфигурации, которые в настоящее время не загружены.\n"
+	.."Их можно безопасно удалить."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "Удалить данные конфигурации"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "запись конфигурации была удалена."
+L["TITAN_PANEL_CHARS"] = "Персонажи"
+L["TITAN_PANEL_CHARS_DESC"] = "Тут есть персонажы с данными настройки."
+L["TITAN_PANEL_REGISTER_START"] = "Регистрация Titan плагинов..."
+L["TITAN_PANEL_REGISTER_END"] = "Процесс регистрации завершен."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Используйте: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffffСбрасывает все на стандартные значения/позиции.";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffСбрасывает масштаб шрифта подсказки панели на стандартное значение.";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffСбрасывает прозрачность подсказки панели на стандартное значение.";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffСбрасывает масштаб на стандартное значение.";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffСбрасывает расстояние кнопок  на стандартное значение.";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Используйте: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffffОткрывает интерфейс контроля панели.";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffffОткрывает интерфейс контроля прозрачности.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffffОткрывает интерфейс контроля шкурок.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Используйте: |cffffffff/titan {исп. профиль <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."исп. профиля <имя> <сервер>: |cffffffffУстанавливает используемые плагины в сохраненный профиль.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<имя>: |cffffffffможет быть задано имя персонажа или название пользовательского профиля."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<сервер>: |cffffffffможет быть задано название сервера или 'TitanCustomProfile'."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffToggles "..TITAN_PANEL.." to load silently.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Используйте: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Используйте: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE.."Масштаб шрифта подсказки Titan Panel сброшен.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE.."Прозрачность шрифта подсказки Titan Panel сброшена.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE.."Масштаб Titan Panel сброшен.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE.."Расстояние кнопок Titan Panel сброшено.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Основной";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Основной профиль";
+L["TITAN_PANEL_GLOBAL_USE"] = "Исп. основной профиль";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Исп. как основной профиль";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Использовать основной профиль для всех персонажей";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "перенастройка опций";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "Вы не можете загрузить профиль, когда используется основной профиль";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffffd700 от |cffff8c00Titan Development Team";
+L["TITAN_PANEL_MENU_TITLE"] = "Титан панель";
+L["TITAN_PANEL_MENU_HIDE"] = "Скрыть";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(В Бою)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(Перезагрузить ПИ)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "Показывать цветной текст";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "Показывать иконку";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "Текст ярлыка";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "Авто-скрытие";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "Текст в центре";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Показать панель";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "Выключить подстройку экрана";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "Отключить подстройку мини-карты";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "Авто подстройка журнала";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "Авто подстройка сумок";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "Авто подстройка  запросов";
+L["TITAN_PANEL_MENU_PROFILES"] = "Профиля";
+L["TITAN_PANEL_MENU_PROFILE"] = "Профиль ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "Выборочный";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " удален.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "Сервер";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "Персонаж";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "Ваш UI будет перезагружен после нажатия 'Okay' для сохранения вашего профиля.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "Введите имя вашего профиля:\n(Макс 20 букв, пропуск недопустим)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "Настройки текущей панели будут сохранены под названием профиля: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "Введенное имя профиля уже существует. Пожалуйста введите уникальное имя.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "Управление";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "Загрузить настройки";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "Удалить";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "Сохранить";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "Конфигурация";
+L["TITAN_PANEL_OPTIONS"] = "Опции";
+L["TITAN_PANEL_MENU_TOP"] = "Верх"
+L["TITAN_PANEL_MENU_TOP2"] = "Верх 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "Низ"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "Низ 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = "Подсказки и фреймы Титан панели";
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Подсказки и фреймы";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Верхняя панель"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Нижняя панель"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "Панели";
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = "Панели Титана";
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = "Дополнительные панели";
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Подсказки";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "Фреймы";
+L["TITAN_PANEL_MENU_PLUGINS"] = "Плагины";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "Блокировать кнопки";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "Показывать версии плагинов";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "Плагин справа";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "Направить модули запусков в правую сторону";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"Сборки титана","Общее","Бой","Информация","Интерфейс","Профессия"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "Показывать подсказки";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "Скрыть подсказки в бою";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Закрепить авто-скрытие в бою";
+L["TITAN_PANEL_MENU_RESET"] = "Сброс панели на стандарт";
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "Настройки шкурки";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "Шрифт панели"
+L["TITAN_PANEL_MENU_ENABLED"] = "Включен";
+L["TITAN_PANEL_MENU_DISABLED"] = "Отключен";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Сдвиг в лева";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Сдвиг в права";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Показать текст плагина";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Всегда вкл";
+L["TITAN_PANEL_MENU_POSITION"] = "Позиция";
+L["TITAN_PANEL_MENU_BAR"] = "Панель";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Показать на панели";
+L["TITAN_PANEL_MENU_SHOW"] = "Показать плагин";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Обновить плагины";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Обновить позицию и текст плагинов";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Silent Load";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Версия";
+L["TITAN_ABOUT_AUTHOR"] = "Автор";
+L["TITAN_ABOUT_CREDITS"] = "Исполнители";
+L["TITAN_ABOUT_CATEGORY"] = "Категория";
+L["TITAN_ABOUT_EMAIL"] = "Email";
+L["TITAN_ABOUT_WEB"] = "Сайт";
+L["TITAN_ABOUT_LICENSE"] = "Лицензия";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "Аддон отображения полосы информации. Позволяет пользователям добавлять вывод данных или плагины модуля запуска на верхнюю панель или нижнюю.";
+L["TITAN_TRANS_MENU_TEXT"] = "Прозрачность";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "Прозрачность";
+L["TITAN_TRANS_MENU_DESC"] = "Регулировка прозрачности панели Титана и подсказок.";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "Главная панель";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "Дополнительная панель";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "Подсказка";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "Регулировка прозрачности подсказки различных плагинов.";
+L["TITAN_UISCALE_MENU_TEXT"] = "Масштаб и шрифт";
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Масштаб и шрифт";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "Размер интерфейса";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "Размер панели";
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "Промежуток кнопок";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Промежуток иконок";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "Размер шрифта подсказок";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "Отключить масштабирование шрифта подсказок";
+L["TITAN_UISCALE_MENU_DESC"] = "Управление различными аспектами панели и интерфейса.";
+L["TITAN_UISCALE_SLIDER_DESC"] = "Регулировка масштаба вашего интерфейса.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "Регулировка масштаба различных кнопок и иконок панели.";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "Регулировка пространства между плагинами с левой стороны.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Регулировка пространства между плагинами с правой стороны  стороне плагинов.Adjusts the space between right-side plugins.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "Регулировка размера подсказки различных плагинов.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "Отключить контроль масштабирования шрифта подсказок Титана.";
+
+L["TITAN_SKINS_TITLE"] = "Шкурки";
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Шкурки - Свои";
+L["TITAN_SKINS_TITLE_CUSTOM"] = "Свои шкурки";
+L["TITAN_SKINS_MAIN_DESC"] = "Все пользовательские шкурки находятся в: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ ".."\n"
+			.."\nшкурки Титана и пользовательские, хранятся в папке Custom."
+L["TITAN_SKINS_LIST_TITLE"] = "Список шкурок";
+L["TITAN_SKINS_SET_DESC"] = "Выберите шкурку для панелей Титана.";
+L["TITAN_SKINS_SET_HEADER"] = "Установить шкурку панели";
+L["TITAN_SKINS_RESET_HEADER"] = "Сброс шкурки Титан панели";
+L["TITAN_SKINS_NEW_HEADER"] = "Добавить новую шкурку";
+L["TITAN_SKINS_NAME_TITLE"] = "Название шкурки";
+L["TITAN_SKINS_NAME_DESC"] = "Введите название вашей новой шкурки.";
+L["TITAN_SKINS_PATH_TITLE"] = "Путь к шкурке";
+L["TITAN_SKINS_PATH_DESC"] = "Введите точный путь к месту где расположены рисунки вашей шкурки, как показано в примере и в объяснении 'Совет'.";
+L["TITAN_SKINS_ADD_HEADER"] = "Добавить шкурку";
+L["TITAN_SKINS_ADD_DESC"] = "Добавить новую шкурку, в список доступных шкурок панели.";
+L["TITAN_SKINS_REMOVE_HEADER"] = "Удалить шкурку";
+L["TITAN_SKINS_REMOVE_DESC"] = "Выберите шкурку для удаления, из доступных шкурок панели.";
+L["TITAN_SKINS_REMOVE_BUTTON"] = "Удалить";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "Удалить выбранную шкурку, из списка доступных шкурок панели.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "Вы несете ответственность за удаление любых нежелательных шкурок "
+	.."из корневой папки Титан Панели. Аддон не может добавлять или удалять файлы."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "Сброс на стандартные значения";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "Сбросить список шкурок на стандартное значение.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "Выберите тип шрифтов для различных плагинов на Титан панеле.";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "Размер шрифта";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "Установка размера шрифта панели.";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = "Слои фрейма панели";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "Регулировка слоя фрейма панелей Титана.";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Дополнительно";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Изменить таймеры, только если у вас возникли проблемы с подстройкой фреймов.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Вход в мир";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Изменить значение (обычно повысить) если фреймы не подстраиваются при входе / выходе в мир или подземелье.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Транспорт";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Изменить значение (обычно повысить) если фреймы не подстраиваются при входе / выходе из транспорта.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "Авто-скрытие панели вкл/выкл";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "Сумки: ";
+L["TITAN_BAG_TOOLTIP"] = "Использование сумок";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "ЛКМ -открыть все сумки";
+L["TITAN_BAG_MENU_TEXT"] = "Сумки";
+L["TITAN_BAG_USED_SLOTS"] = "Используемые слоты";
+L["TITAN_BAG_FREE_SLOTS"] = "Свободные слоты";
+L["TITAN_BAG_BACKPACK"] = "Рюкзак";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "Показывать исп. сумки";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "Показывать доступные слоты";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "Отображать подробную подсказку";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "Игнорировать сумки";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "Игнорировать сумки для профф";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Зачарованный мешочек из магической ткани", "Зачарованная сумка из рунической ткани", "Сумка зачаровывателя", "Большая зачарованная сумка", "Сумка из огненной чароткани", "Таинственная котомка",
+"Трансцендентная сумка", "\"Дилижанс - Эксклюзив\" - очаровательный коктейльный ридикюль"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Тяжелый ящик с инструментами", "Ящик для инструментов из оскверненного железа", "Титановый ящик с инструментами", "Кориевый ящик с инструментами",  "\"Дилижанс - Мэдди\" - высокотехнологичная сумка"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Мешочек для трав", "Кенарийская сумка для трав", "Сумка Кенария", "Ботаническая сумка Микаа", "Изумрудный вьюк", "Сумка хиджальской экспедиции",
+"\"Дилижанс - Зеленые\" - сумка для перевозки трав"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Сумка начертателя", "Рюкзак с бесчисленными карманами", "\"Дилижанс - Ксандера\" - школьный рюкзак", "Burnished Inscription Bag"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Мешочек для самоцветов", "Мешочек для драгоценностей", "\"Дилижанс - Эксклюзив\" - украшенный самоцветами клатч"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Сумка кожевника", "Сумка множества шкур", "Походная сума зверолова", "\"Дилижанс - Мия\" - кожаная сумка", "Burnished Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Шахтерский мешок", "Укрепленная шахтерская сумка", "Шахтерская сумка из шкуры мамонта", "\"Дилижанс - Кристина\" - дорогостоящая металлическая сумка", "Burnished Mining Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Ящик для рыболовной снасти мастера приманки"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Portable Refrigerator", "Advanced Refrigeration Unit"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "Часы";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "Часовое смещение: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "Местное время: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "Серверное время: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "Заданное серверное время: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "ЛКМ для настройки разницы с ОФФ"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "12/24 ч формат";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "[Shift Левый-Клик] открывает календарь.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "Часовое смещение: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "Смещение";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "Фрмт 24ч";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "Переключение между отображением времени в 12-часовым и 24-часовым форматом";
+L["TITAN_CLOCK_MENU_TEXT"] = "Часы";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "Отображать местное время(М)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "Отображать серверное время (С)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "Отображать заданное серверное время (З)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "Отображать часы справа";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "Скрыть кнопку времени/календаря";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Скрыть кнопку времени";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Скрыть кнопку календаря";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "Формат координат";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "Место: ";
+L["TITAN_LOCATION_TOOLTIP"] = "Информация о локации";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "Shift + ЛКМ для добавления локации";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "Информации в чат";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "Зона: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "Подзона: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "PVP инфо: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "Место дома";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "Дом: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "Координаты";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "Показывать зону";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "Показывать координаты на мировой карте";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "Курсор(X,Y): %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "Игрок(X,Y): %s";
+L["TITAN_LOCATION_NO_COORDS"] = "Нет координат";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "Показывать местонахождения на мини-карте";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Обновить карту мира при смене зоны";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "FPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "Текущий FPS: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "Средний FPS: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "Мин. FPS: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "Макс. FPS: ";
+L["TITAN_FPS_TOOLTIP"] = "Кадров в секунду";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."мс";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."КБ/с";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "Задержка: ";
+L["TITAN_LATENCY_TOOLTIP"] = "Статус сети";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Задержка мира (дом): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Задержка игры (мир): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "Вход пропуск.способность: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "Выход пропуск.способность: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Задержка";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "Обыск: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "Каждый за себя";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "По очереди";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "Ответственный за добычу";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "Групповая очередь";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "Приоритет по нужности";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Персонализированная";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "Информация о типе добычи";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "Тип добычи";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "Случайный бросок костей";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "Совет: ЛКМ для броска костей.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "Совет: ПКМ для выбора типа броска костей из меню.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "Сложность подземелья";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "Сложность рейда";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "Сложность рейда/подземелья";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "Установить сложность подземелья";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "Установить сложность рейда";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "Авто (пологаясь на группу)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."мб";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."КБ/с";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "Память: ";
+L["TITAN_MEMORY_TOOLTIP"] = "Использование памяти";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "Текущая: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "Исходная: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "Темп прироста: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "Производительность";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "Производительность";
+L["TITAN_PERFORMANCE_ADDONS"] = "Аддоны используют";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "Использование памети аддонами";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "Формат памяти аддонов";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "Загрузка ЦПУ аддонами";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "Имя:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "Испл";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "Скорость";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "Используется памяти аддонами:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "Использование CPU:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "Показывать FPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "Показывать задержку мира";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Показывать задержку игры";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "Показывать память";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "Используемая память аддонами";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "Показ размер исп. аддонами";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "Режим анализа ЦПУ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "Включить режим анализа ЦПУ ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "Отключить режим анализа ЦПУ ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "Мониторинг Аддонов: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "Мониторинг Аддонов";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "Совет: Для сжатия/очистки памяти нажмите ЛКМ.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "ОПТ/ч на этом уровне: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "ОПТ/ч за эту сессию: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "Время до уровня: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "Завершение уровня: ";
+L["TITAN_XP_TOTAL_RESTED"] = "Отдых: ";
+L["TITAN_XP_XPTOLEVELUP"] = "Опыта до поднятия урв: ";
+L["TITAN_XP_TOOLTIP"] = "Информация о опыте";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "Всего времени сыграно: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "Время игры на этом уровне: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "Время сыграно за эту сессию: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "Всего Опыта на этот уровень: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "Опыта получено на этом уровне: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "Опыта нужно для уровня: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "Опыта получено за эту сессию: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "Опыт/ч на этом уровне: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "Опыт/ч за эту сессию: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "Время до уровня (темп уровня): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "Время до уровня (темп сессии): ";
+L["TITAN_XP_MENU_TEXT"] = "Опыт";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "Показывать Опыт/ч на этом уровне";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "Показывать Опыт/ч за эту сессию";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Вид мульти-инфо";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "Кнопка";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "Показ опыта за отдых";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "Показ опыта до уровня";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "Показ прим.убийств до уровня";
+L["TITAN_XP_MENU_RESET_SESSION"] = "Сбросить сессию";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "Обновить таймеры";
+L["TITAN_XP_UPDATE_PENDING"] = "Обновляется...";
+L["TITAN_XP_KILLS_LABEL"] = "Убийств до уровня (%s опыта полученно): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "Оц. убийств: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "Время сеанса: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "Показать время сеанса";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*) погибает, вы получаете (%d+) |4очко:очка:очков; опыта.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "Получение опыта: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "Опыта до уровня (%s опыта в последний раз): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "Показать необходимый опыт до уровня";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "Ремонт",
+	tooltip = "Информация о прочности амуниции",
+	button = "Прочность: ",
+	normal = "Стоимость ремонта (Стандарт): ",
+	friendly = "Стоимость ремонта (Дружелюбие): ",
+	honored = "Стоимость ремонта (Уважение): ",
+	revered = "Стоимость ремонта (Почтение): ",
+	exalted = "Стоимость ремонта (Превознесение): ",
+	buttonNormal = "Показать обычное",
+	buttonFriendly = "Показать Дружелюбие (5%)",
+	buttonHonored = "Показать Уважение (10%)",
+	buttonRevered = "Показать Почтение (15%)",
+	buttonExalted = "Показать Превознесение (20%)",
+	percentage = "Показывать в процентах",
+	itemnames = "Показывать название предмета",
+	mostdamaged = "Показывать наибольше повреждённую",
+	showdurabilityframe = "Показывать фрейм прочности",
+	undamaged = "Показывать неповреждённые предметы",
+	discount = "Скидка",
+	nothing = "Нет поврежденных вещей.",
+	confirmation = "Вы хотите отремонтировать все одетые предметы?",
+	badmerchant = "Этот торговец не может ремонтировать.",
+	popup = "Показать всплывающее меню ремонта",
+	showinventory = "Подсчет повреждение инвентаря",
+	WholeScanInProgress = "Обновление...",
+	AutoReplabel = "Авто-Ремонт",
+	AutoRepitemlabel = "Авто-Ремонт всех предметов",
+	ShowRepairCost = "Отображать стоимость ремонта",
+	ignoreThrown = "Игнорировать метательное оружые",
+	ShowItems = "Показать предметы",
+	ShowDiscounts = "Показать скидки",
+	ShowCosts = "Показать стоимость",
+	Items = "Предметы",
+	Discounts = "Скидки",
+	Costs = "Стоимость",
+	CostTotal = "Всего",
+	CostBag = "Того что в сумках",
+	CostEquip = "Того что на персонаже",
+	TooltipOptions = "Подсказка",
+};
+
+L["TITAN_REPAIR"] = "Titan - ремонт"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "Средства гильд банка :"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "Доступная сумма из гильд банка :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "Использовать средства гильд банка"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "Гильд Банк не в состоянии заплотить за ремонт, или вы не можете воспользоваться такой суммой."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "Вы не состоите в гильдии либо у вас нет полномочий использовать средства с гильд банка для ремонта ваших предметов."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "В данный момент вы не в состоянии отремонтироваться."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "Сообщить цену ремонта в чат"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "Ремонт обошелся в "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "Всего золота";
+L["TITAN_GOLD_ITEMNAME"] = "Слежение за Золотом";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "Очистить базу данных";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "Сбросить текущий сеанс";
+L["TITAN_GOLD_DB_CLEARED"] = "Титан: Слежение за Золотом - База данных очищена.";
+L["TITAN_GOLD_SESSION_RESET"] = "Титан: Слежение за Золотом - Сеан сброшен.";
+L["TITAN_GOLD_MENU_TEXT"] = "Слежение за Золотом";
+L["TITAN_GOLD_TOOLTIP"] = "Информация о золоте";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "Отображать золото у игрока";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "Отображать золото на сервера";
+L["TITAN_GOLD_SESS_EARNED"] = "Заработано за данный сеанс";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "Заработано за час";
+L["TITAN_GOLD_SESS_LOST"] = "Потрачено за данный сеанс";
+L["TITAN_GOLD_PERHOUR_LOST"] = "Потрачено за час";
+L["TITAN_GOLD_STATS_TITLE"] = "Статистика сеанса";
+L["TITAN_GOLD_TTL_GOLD"] = "Всего золота";
+L["TITAN_GOLD_START_GOLD"] = "Начальное количество золота";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "Сортировать по золоту";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "Сортировать по имени";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "Отображать золото за час";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "Скрыть золота за час";
+L["TITAN_GOLD_GOLD"] = "з";
+L["TITAN_GOLD_SILVER"] = "с";
+L["TITAN_GOLD_COPPER"] = "м";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "видимый";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "скрыт";
+L["TITAN_GOLD_DELETE_PLAYER"] = "Удалить персонажа";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Show toon";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "Альянс";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "Орда";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."Внимание: "
+..FONT_COLOR_CODE_CLOSE.."Данное действие уничтожит вашу базу данных Titan Gold. "
+.."Если вы хотите продолжить, жмите 'Принять', если нет, жмите 'Отмена' или клавишу 'Escape'.";
+L["TITAN_GOLD_COIN_NONE"] = "Без меток";
+L["TITAN_GOLD_COIN_LABELS"] = "Показать метки денег";
+L["TITAN_GOLD_COIN_ICONS"] = "Показать иконки";
+L["TITAN_GOLD_ONLY"] = "Показать только золото";
+L["TITAN_GOLD_COLORS"] = "Показать окраску золота";
+L["TITAN_GOLD_MERGE"] = "Merge Servers";
+L["TITAN_GOLD_SEPARATE"] = "Separate Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "Контроль звука";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "Громкость общего звука: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "Громкость звуковых эффектов: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "Громкость звуков окружающего мира: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "Громкость мызыки: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "Громкость микрофона: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "Громкость динамиков: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "ЛКМ чтобы настроить"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "Громкость звука";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "Громкость звука: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "Громкость";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "Общий";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "Звуки";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "Мир";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "Музыка";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "Микрафон";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "Динамик";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "Макс";
+L["TITAN_VOLUME_CONTROL_LOW"] = "Мин";
+L["TITAN_VOLUME_MENU_TEXT"] = "Настройки Звука";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "Настройки звука/голоса" ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "Перезаписать звуковые настройки близзарда";
diff --git a/Titan/locale/Localization.TW.lua b/Titan/locale/Localization.TW.lua
new file mode 100644
index 0000000..f879a29
--- /dev/null
+++ b/Titan/locale/Localization.TW.lua
@@ -0,0 +1,553 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","zhTW")
+if not L then return end
+
+L["TITAN_PANEL"] = "Titan Panel";
+local TITAN_PANEL = "Titan Panel";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/A";
+L["TITAN_SECONDS"] = "秒鐘";
+L["TITAN_MINUTES"] = "分鐘";
+L["TITAN_HOURS"] = "小時";
+L["TITAN_DAYS"] = "天";
+L["TITAN_SECONDS_ABBR"] = "秒";
+L["TITAN_MINUTES_ABBR"] = "分";
+L["TITAN_HOURS_ABBR"] = "時";
+L["TITAN_DAYS_ABBR"] = "天";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "KB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kbps"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "無";
+L["TITAN_USE_COMMA"] = "Use comma";
+L["TITAN_USE_PERIOD"] = "Use period";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "你可能已經刪除目前的角色配置。";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."Warning : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "如果你要繼續請按「套用」,你的插件將會重新載入,如果不確定請按「取消」或 Esc 鍵。"
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."此設定將會重置你的 Titan 的設定為預設值,並且會重新建立你的配置檔案。 "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."這會重新載入泰坦面板. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = TITAN_PANEL.." Attempts"
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Attempts"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "The plugins below requested to be registered with "..TITAN_PANEL..".\n"
+	.."Please send any issues to the plugin author."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Type"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Category"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Button Name"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Status"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Issue"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Notes"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Table index"
+L["TITAN_PANEL_EXTRAS"] = TITAN_PANEL.." Extras"
+L["TITAN_PANEL_EXTRAS_SHORT"] = "Extras"
+L["TITAN_PANEL_EXTRAS_DESC"] = "These are plugins with configuration data that are not currently loaded.\n"
+	.."These are safe to delete."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "Delete config data"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "configuration entry has been removed."
+L["TITAN_PANEL_CHARS"] = "Characters"
+L["TITAN_PANEL_CHARS_DESC"] = "These are characters with configuration data."
+L["TITAN_PANEL_REGISTER_START"] = "Register "..TITAN_PANEL.." plugins..."
+L["TITAN_PANEL_REGISTER_END"] = "Registration process done."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."輸入: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffff重置面板為預設值/位置。";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffff重置面板提示文字大小為預設值。";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffff重製面板提示視窗透明度為預設值。";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffff重置面板大小為預設值。";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffff重置面板按鈕間距為預設值。";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffff開啟 Ace3 面板控制介面。";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffff開啟 Ace3 透明度控制介面。";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffff開啟 Ace3 外觀控制介面。";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <name> <server>: |cffffffffSets the profile to the requested saved profile.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<name>: |cffffffffcan be either the character name or the custom profile name."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<server>: |cffffffffcan be either the server name or 'TitanCustomProfile'."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffToggles "..TITAN_PANEL.." to load silently.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel 提示文字大小已重置。";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel 提示視窗透明度已重置。";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel 大小已重置。";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE.."Titan Panel 按鈕間距已重置。";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Global";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Global Profile";
+L["TITAN_PANEL_GLOBAL_USE"] = "Use Global Profile";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Use as Global Profile";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Use a global profile for all characters";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "resetting options";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "You may not load a profile when a global profile is in use";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffff8c00Titan Development Team |cffffffff Presents ".." |cffffffff";
+L["TITAN_PANEL_MENU_TITLE"] = "泰坦面版";
+L["TITAN_PANEL_MENU_HIDE"] = "隱藏";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(戰鬥中)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(重載介面)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "顯示彩色文字";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "顯示圖示";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "顯示標籤文字";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "自動隱藏";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "文字置中";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Show Bar";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "關閉螢幕調整";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "關閉小地圖調整";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "自動調整紀錄";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "自動調整背包";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "自動調整標籤框體";
+L["TITAN_PANEL_MENU_PROFILES"] = "配置檔";
+L["TITAN_PANEL_MENU_PROFILE"] = "配置檔「";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "自訂";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = "」已刪除。";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "伺服器";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "角色";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "當按下「確定」按鈕時,你的插件將立即新載入並儲存你的自訂配置檔。";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "請輸入你的自訂配置檔名稱:\n(最大 20 個字元,不可有空格且區分大小寫。)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "目前的面板設定將會儲存到配置檔: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "此配置檔名稱已存在,你確定你要覆蓋嗎?如果確定請按「套用」,否則請按「取消」或 ESC 鍵。";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "管理";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "載入設定";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "刪除";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "儲存";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "配置設定";
+L["TITAN_PANEL_OPTIONS"] = "選項";
+L["TITAN_PANEL_MENU_TOP"] = "Top"
+L["TITAN_PANEL_MENU_TOP2"] = "Top 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "Bottom"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "Bottom 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = TITAN_PANEL.." Tooltips and Frames";
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Tooltips and Frames";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Top Bars"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Bottom Bars"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "狀態列";
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = TITAN_PANEL.." Top Bars";
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = TITAN_PANEL.." Bottom Bars";
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Tooltips";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "視窗";
+L["TITAN_PANEL_MENU_PLUGINS"] = "插件";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "鎖定面板";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "顯示插件版本";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "右側插件";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "強制快捷列於右側";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"內建插件","一般","戰鬥","資訊","介面","專業"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "顯示提示";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "戰鬥中隱藏提示訊息";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Lock auto hide bars while in combat";
+L["TITAN_PANEL_MENU_RESET"] = "重置面板為預設值";
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "面板設定";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "面版字體"
+L["TITAN_PANEL_MENU_ENABLED"] = "啟用";
+L["TITAN_PANEL_MENU_DISABLED"] = "停用";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Shift 左鍵";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Shift 右鍵";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Show plugin text";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Always on";
+L["TITAN_PANEL_MENU_POSITION"] = "Position";
+L["TITAN_PANEL_MENU_BAR"] = "Bar";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Display on Bar";
+L["TITAN_PANEL_MENU_SHOW"] = "Show plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Refresh plugins";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Refresh plugin text and position";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Silent Load";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Version";
+L["TITAN_ABOUT_AUTHOR"] = "Author";
+L["TITAN_ABOUT_CREDITS"] = "Credits";
+L["TITAN_ABOUT_CATEGORY"] = "Category";
+L["TITAN_ABOUT_EMAIL"] = "Email";
+L["TITAN_ABOUT_WEB"] = "Website";
+L["TITAN_ABOUT_LICENSE"] = "License";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "顯示狀態列插件資訊。允許使用者在螢幕控制面板的上方或下方加入資料或快捷列插件。";
+L["TITAN_TRANS_MENU_TEXT"] = TITAN_PANEL.." Transparency";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "透明度";
+L["TITAN_TRANS_MENU_DESC"] = "調整 Titan 狀態列和提示視窗的透明度。";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "主要狀態列";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "輔助狀態列";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "提示訊息";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "設定各種提示訊息插件的透明度。";
+L["TITAN_UISCALE_MENU_TEXT"] = "面板控制";
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Scale and Font";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "插件大小";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = "面板大小";
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "按鈕間距";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Icon Spacing";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "提示訊息文字大小";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "關閉提示訊息文字大小";
+L["TITAN_UISCALE_MENU_DESC"] = "控制各種插件和面板的外觀。";
+L["TITAN_UISCALE_SLIDER_DESC"] = "設定整體插件的大小。";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "設定各種面板按鈕及圖示大小。";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "調整左側插件間的空間大小";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Adjusts the space between right-side plugins.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "調整各種提示訊息插件的大小。";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "關閉 Titan 的提示訊息字型大小控制。";
+
+L["TITAN_SKINS_TITLE"] = TITAN_PANEL.." Skins";
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Skins - Custom";
+L["TITAN_SKINS_TITLE_CUSTOM"] = TITAN_PANEL.." Custom Skins";
+L["TITAN_SKINS_MAIN_DESC"] = "All custom skins are assumed to be in: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ ".."\n"
+			.."\n"..TITAN_PANEL.." and custom skins are stored under the Custom folder."
+L["TITAN_SKINS_LIST_TITLE"] = "外觀清單";
+L["TITAN_SKINS_SET_DESC"] = "選擇一個 Titan 狀態列的外觀設定。";
+L["TITAN_SKINS_SET_HEADER"] = "設定面板外觀";
+L["TITAN_SKINS_RESET_HEADER"] = "Reset "..TITAN_PANEL.." Skins";
+L["TITAN_SKINS_NEW_HEADER"] = "加入新外觀";
+L["TITAN_SKINS_NAME_TITLE"] = "外觀名稱";L["TITAN_SKINS_NAME_DESC"] = "請輸入新外觀的名稱。";
+L["TITAN_SKINS_NAME_DESC"] = "例如: My Titan Skin";
+L["TITAN_SKINS_PATH_TITLE"] = "外觀路徑";
+L["TITAN_SKINS_PATH_DESC"] = "請輸入放置外觀檔案的正確路徑,如範例所示,說明請看「注意」事項。";
+L["TITAN_SKINS_ADD_HEADER"] = "新增外觀";
+L["TITAN_SKINS_ADD_DESC"] = "新增一個面板可用的外觀到清單中。";
+L["TITAN_SKINS_REMOVE_HEADER"] = "移除外觀";
+L["TITAN_SKINS_REMOVE_DESC"] = "從清單中選擇欲移除的面板外觀。";
+L["TITAN_SKINS_REMOVE_BUTTON"] = "移除";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "從清單中移除所選的面板外觀。";
+L["TITAN_SKINS_REMOVE_NOTES"] = "You are responsible for removing any unwanted custom skins "
+	.."from the "..TITAN_PANEL.." install folder. Addons can not add or remove files."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "重置配置";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "重置外觀列表至默認值.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "選擇泰坦條上各個模組的字體樣式.";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "字體大小";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "設置泰坦面板的字體大小.";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = "面版框架層疊";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "為泰坦條設置是否顯示在最前.";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Advanced";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Change Timers only if you experience issues with frames not adjusting.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Entering World";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving world or an instance.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Vehicle";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving vehicle.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "自動隱藏 開/關";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "背包: ";
+L["TITAN_BAG_TOOLTIP"] = "背包";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "提示:滑鼠左鍵打開所有背包";
+L["TITAN_BAG_MENU_TEXT"] = "背包";
+L["TITAN_BAG_USED_SLOTS"] = "已使用空間";
+L["TITAN_BAG_FREE_SLOTS"] = "閒置空間";
+L["TITAN_BAG_BACKPACK"] = "背包";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "顯示已使用空間";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "顯示剩餘空間";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "顯示詳細提示訊息";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "忽略容器";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "忽略專業背包空間";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Enchanted Mageweave Pouch", "Enchanted Runecloth Bag", "Enchanter's Satchel", "Big Bag of Enchantment", "Spellfire Bag",
+"Mysterious Bag", "Otherworldly Bag", "\"Carriage - Exclusive\" Enchanting Evening Purse"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Heavy Toolbox", "Fel Iron Toolbox", "Titanium Toolbox", "Khorium Toolbox", "Elementium Toolbox", "\"Carriage - Maddy\" High Tech Bag"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Herb Pouch", "Cenarion Herb Bag", "Satchel of Cenarius", "Mycah's Botanical Bag", "Emerald Bag", "Hyjal Expedition Bag",
+"\"Carriage - Going Green\" Herb Tote Bag"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Scribe's Satchel", "Pack of Endless Pockets", "\"Carriage - Xandera\" Student's Satchel", "Burnished Inscription Bag"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Gem Pouch", "Bag of Jewels", "\"Carriage - Exclusive\" Gem Studded Clutch"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Leatherworker's Satchel", "Bag of Many Hides", "Trapper's Traveling Pack", "\"Carriage - Meeya\" Leather Bag", "Burnished Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Mining Sack", "Reinforced Mining Bag", "Mammoth Mining Bag", "\"Carriage - Christina\" Precious Metal Bag", "Burnished Mining Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Lure Master Tackle Box"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Portable Refrigerator", "Advanced Refrigeration Unit"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "時鐘";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "時差值: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "本地時間: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "伺服器時間: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "伺服器調整時間: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "提示:滑鼠左鍵調整時間"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "切換 12/24 小時制";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Shift+左鍵開啟/關閉行事曆。";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "時差: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "時差";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24 小時";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "切換 12/24 小時制";
+L["TITAN_CLOCK_MENU_TEXT"] = "時鐘";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "顯示本地時間 (L)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "顯示伺服器時間 (S)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "顯示伺服器調整時間 (A)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "顯示靠最右邊"
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "隱藏 時間/日曆 按鈕";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Hide Time Button";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Hide Calendar Button";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "座標格式";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "座標: ";
+L["TITAN_LOCATION_TOOLTIP"] = "座標資訊";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "提示:Shift+滑鼠左鍵來增加位置資訊";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "顯示資訊在頻道上";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "區域: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "地區: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "PVP 資訊: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "爐石設定位置";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "旅館: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "位置";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "顯示區域文字";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "顯示座標在世界地圖上";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "游標(X,Y): %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "玩家(X,Y): %s";
+L["TITAN_LOCATION_NO_COORDS"] = "無座標";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "在小地圖上顯示位置";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Update World Map When Zone Changes";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "FPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "目前 FPS: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "平均 FPS: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "最小 FPS: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "最大 FPS: ";
+L["TITAN_FPS_TOOLTIP"] = "每秒的畫面數";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "連線速度: ";
+L["TITAN_LATENCY_TOOLTIP"] = "網路狀態";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Realm Latency (home): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Game Latency (world): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "頻寬(IN): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "頻寬(OUT): ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Latency";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "拾取方式: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "自由拾取";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "輪流拾取";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "隊長分配";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "團隊分配";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "需求分配";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "拾取方式";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "拾取方式";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "隨機擲骰";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "提示: 左鍵隨機擲骰。";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "右鍵從選單中選擇擲骰類型。";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "五人副本難度";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "團隊副本難度";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "顯示副本難度";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "設置五人副本難度";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "設置團隊副本難度";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "自動設定 (基於隊伍類型)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "記憶體: ";
+L["TITAN_MEMORY_TOOLTIP"] = "記憶體使用狀態";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "目前: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "初始: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "增加比率: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "畫面表現";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "畫面表現";
+L["TITAN_PERFORMANCE_ADDONS"] = "插件使用情況";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "插件記憶體使用情況";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "插件記憶體格式";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "插件 CPU 使用情況";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "名稱: ";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "使用情況";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "比率";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "總插件使用記憶體 :";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "總插件使用 CPU :";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "顯示 FPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "顯示連線速度";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Show Game Latency";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "顯示記憶體";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "顯示插件記憶體使用情況";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "顯示插件使用比率";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "CPU 配置模式";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "啟用 CPU 配置模式 ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "關閉 CPU 配置模式 ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "已監督插件: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "已監督插件";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "提示: 左鍵強制清除垃圾。";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "EXP/小時 目前等級: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "EXP/小時 目前階段: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "升級所需的時間: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "等級完成: ";
+L["TITAN_XP_TOTAL_RESTED"] = "休息: ";
+L["TITAN_XP_XPTOLEVELUP"] = "等級提昇尚須經驗值: ";
+L["TITAN_XP_TOOLTIP"] = "EXP 經驗值資訊";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "總共遊戲時間: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "這個等級遊戲的時間: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "這個階段遊戲的時間: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "這個等級總共要求的 EXP: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "這個等級內所獲得的 EXP: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "升級下一級所需要的 EXP: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "這個階段內所獲得的 EXP: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "EXP/小時 這個等級內: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "EXP/小時 這個階段內: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "至升級所需的時間 (等級比率): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "至升級所需的時間 (階段比率): ";
+L["TITAN_XP_MENU_TEXT"] = "經驗值系統(EXP)";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "顯示 EXP/小時 這個等級";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "顯示 EXP/小時 這個階段";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "顯示升級尚須的休息/EXP";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "按鈕";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "顯示休息的經驗值";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "顯示升級還需要的 EXP";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "顯示升級還需要的預計擊殺數";
+L["TITAN_XP_MENU_RESET_SESSION"] = "重設經驗值統計";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "更新時間";
+L["TITAN_XP_UPDATE_PENDING"] = "更新中...";
+L["TITAN_XP_KILLS_LABEL"] = "升級還需的擊殺數(上次獲得 %s EXP): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "預估擊殺數: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "連接: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "顯示連接時間";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*)死亡,你獲得(%d+)點經驗。";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "預估獲得經驗: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "升級所需擊殺數(基於最後擊殺獲得的%s點經驗): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "顯示升級所需(基於最後所獲經驗)";
+
+-- Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "裝備耐久度",
+	tooltip = "裝備耐久度",
+	button = "裝備耐久度: ",
+	normal = "修裝花費 (正常): ",
+	friendly = "修裝花費 (友善): ",
+	honored = "修裝花費 (尊敬): ",
+	revered = "修裝花費 (崇敬): ",
+	exalted = "修裝花費 (崇拜): ",
+	buttonNormal = "顯示正常",
+	buttonFriendly = "顯示友善 (5%)",
+	buttonHonored = "顯示尊敬 (10%)",
+	buttonRevered = "顯示崇敬 (15%)",
+	buttonExalted = "顯示崇拜 (20%)",
+	percentage = "顯示百分比",
+	itemnames = "顯示物品的名稱",
+	mostdamaged = "損壞最多的",
+	showdurabilityframe = "顯示耐久面板",
+	undamaged = "顯示未損壞的物品",
+	discount = "顯示折扣於物品名稱和按鈕上",
+	nothing = "沒有損壞的物品",
+	confirmation = "你要修理所有已裝備物品嗎?",
+	badmerchant = "此商人無法修裝。 顯示正常的修裝花費",
+	popup = "顯示修理對話框",
+	showinventory = "計算背包中裝備損壞度",
+	WholeScanInProgress = "更新中...",
+	AutoReplabel = "自動修裝",
+	AutoRepitemlabel = "自動修理所有裝備",
+	ShowRepairCost = "顯示修裝費用",
+	ignoreThrown = "忽略投擲武器",
+	ShowItems = "Show Items",
+	ShowDiscounts = "Show Discounts",
+	ShowCosts = "Show Costs",
+	Items = "Items",
+	Discounts = "Discounts",
+	Costs = "Costs",
+	CostTotal = "Total Cost",
+	CostBag = "Bag Cost",
+	CostEquip = "Equip Cost",
+	TooltipOptions = "Tooltip",
+};
+
+L["TITAN_REPAIR"] = "泰坦修理"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "公會銀行資金 :"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "你可以使用的資金 :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "使用公會資金修理"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "公會資金不足或者你超過了允許的修理費用."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "你沒有公會或者你沒有使用公會資金修理的許可."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "你現在支付不起修理費用."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "輸出修理費用到聊天頻道"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "修理費用是 "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "總計金錢於";
+L["TITAN_GOLD_ITEMNAME"] = "金錢追蹤";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "清除資料庫";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "重置目前階段";
+L["TITAN_GOLD_DB_CLEARED"] = "泰坦金錢追蹤 - 資料庫已清除。";
+L["TITAN_GOLD_SESSION_RESET"] = "泰坦金錢追蹤 - 此階段已重置。";
+L["TITAN_GOLD_MENU_TEXT"] = "金錢追蹤";
+L["TITAN_GOLD_TOOLTIP"] = "金錢資訊";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "顯示玩家賺取金額";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "顯示伺服器賺取金額";
+L["TITAN_GOLD_SESS_EARNED"] = "此階段賺取";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "每小時賺取";
+L["TITAN_GOLD_SESS_LOST"] = "此階段花費";
+L["TITAN_GOLD_PERHOUR_LOST"] = "每小時花費";
+L["TITAN_GOLD_STATS_TITLE"] = "階段統計";
+L["TITAN_GOLD_TTL_GOLD"] = "總計金額";
+L["TITAN_GOLD_START_GOLD"] = "開始統計金額";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "依金額排列表單";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "依名稱排列表單";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "顯示每小時賺取金額";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "隱藏每小時賺取金額";
+L["TITAN_GOLD_GOLD"] = "g";
+L["TITAN_GOLD_SILVER"] = "s";
+L["TITAN_GOLD_COPPER"] = "c";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "顯示";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "隱藏";
+L["TITAN_GOLD_DELETE_PLAYER"] = "刪除角色";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Show toon";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "聯盟";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "部落";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."警告: "
+..FONT_COLOR_CODE_CLOSE.."這會清空金幣助手的資料庫,按取消放弃此次操作.";
+L["TITAN_GOLD_COIN_NONE"] = "Show No Labels";
+L["TITAN_GOLD_COIN_LABELS"] = "Show Text Labels";
+L["TITAN_GOLD_COIN_ICONS"] = "Show Icon Labels";
+L["TITAN_GOLD_ONLY"] = "Show Gold Only";
+L["TITAN_GOLD_COLORS"] = "Show Gold Colors";
+L["TITAN_GOLD_MERGE"] = "Merge Servers";
+L["TITAN_GOLD_SEPARATE"] = "Separate Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "音量控制";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "主音量大小: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "音效音量大小: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "環境音量大小: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "音樂音量大小: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "麥克風音量大小: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "喇叭音量大小: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "提示: 滑鼠左鍵調整"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "主音量大小";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "主音量: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "音量";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "主音量";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "音效";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "環境";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "音樂";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "麥克風";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "喇叭";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "高";
+L["TITAN_VOLUME_CONTROL_LOW"] = "低";
+L["TITAN_VOLUME_MENU_TEXT"] = "音量";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "顯示 聲音選項" ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "替換默認聲音控制";
diff --git a/Titan/locale/Localization.lua b/Titan/locale/Localization.lua
new file mode 100644
index 0000000..952fb6e
--- /dev/null
+++ b/Titan/locale/Localization.lua
@@ -0,0 +1,552 @@
+local L = LibStub("AceLocale-3.0"):NewLocale("Titan","enUS",true)
+
+L["TITAN_PANEL"] = "Titan Panel";
+local TITAN_PANEL = "Titan Panel";
+L["TITAN_DEBUG"] = "<Titan>";
+L["TITAN_PRINT"] = "Titan";
+
+L["TITAN_NA"] = "N/A";
+L["TITAN_SECONDS"] = "seconds";
+L["TITAN_MINUTES"] = "minutes";
+L["TITAN_HOURS"] = "hours";
+L["TITAN_DAYS"] = "days";
+L["TITAN_SECONDS_ABBR"] = "s";
+L["TITAN_MINUTES_ABBR"] = "m";
+L["TITAN_HOURS_ABBR"] = "h";
+L["TITAN_DAYS_ABBR"] = "d";
+L["TITAN_MILLISECOND"] = "ms";
+L["TITAN_KILOBYTES_PER_SECOND"] = "KB/s";
+L["TITAN_KILOBITS_PER_SECOND"] = "kbps"
+L["TITAN_MEGABYTE"] = "MB";
+L["TITAN_NONE"] = "None";
+L["TITAN_USE_COMMA"] = "Use comma";
+L["TITAN_USE_PERIOD"] = "Use period";
+
+L["TITAN_PANEL_ERROR_PROF_DELCURRENT"] = "You may not delete your current profile.";
+local TITAN_PANEL_WARNING = GREEN_FONT_COLOR_CODE.."Warning : "..FONT_COLOR_CODE_CLOSE
+local TITAN_PANEL_RELOAD_TEXT = "If you wish to continue with this operation, push 'Accept' (your UI will reload), otherwise push 'Cancel' or the 'Escape' key."
+L["TITAN_PANEL_RESET_WARNING"] = TITAN_PANEL_WARNING
+	.."This setting will reset your bar(s) and "..TITAN_PANEL.." settings to default values and will recreate your current profile. "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_RELOAD"] = TITAN_PANEL_WARNING
+	.."This will reload "..TITAN_PANEL..". "
+	..TITAN_PANEL_RELOAD_TEXT
+L["TITAN_PANEL_ATTEMPTS"] = TITAN_PANEL.." Attempts"
+L["TITAN_PANEL_ATTEMPTS_SHORT"] = "Attempts"
+L["TITAN_PANEL_ATTEMPTS_DESC"] = "The plugins below requested to be registered with "..TITAN_PANEL..".\n"
+	.."Please send any issues to the plugin author."
+L["TITAN_PANEL_ATTEMPTS_TYPE"] = "Type"
+L["TITAN_PANEL_ATTEMPTS_CATEGORY"] = "Category"
+L["TITAN_PANEL_ATTEMPTS_BUTTON"] = "Button Name"
+L["TITAN_PANEL_ATTEMPTS_STATUS"] = "Status"
+L["TITAN_PANEL_ATTEMPTS_ISSUE"] = "Issue"
+L["TITAN_PANEL_ATTEMPTS_NOTES"] = "Notes"
+L["TITAN_PANEL_ATTEMPTS_TABLE"] = "Table index"
+L["TITAN_PANEL_EXTRAS"] = TITAN_PANEL.." Extras"
+L["TITAN_PANEL_EXTRAS_SHORT"] = "Extras"
+L["TITAN_PANEL_EXTRAS_DESC"] = "These are plugins with configuration data that are not currently loaded.\n"
+	.."These are safe to delete."
+L["TITAN_PANEL_EXTRAS_DELETE_BUTTON"] = "Delete config data"
+L["TITAN_PANEL_EXTRAS_DELETE_MSG"] = "configuration entry has been removed."
+L["TITAN_PANEL_CHARS"] = "Characters"
+L["TITAN_PANEL_CHARS_DESC"] = "These are characters with configuration data."
+L["TITAN_PANEL_REGISTER_START"] = "Register "..TITAN_PANEL.." plugins..."
+L["TITAN_PANEL_REGISTER_END"] = "Registration process done."
+
+-- slash command help
+L["TITAN_PANEL_SLASH_RESET_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {reset | reset tipfont/tipalpha/panelscale/spacing}";
+L["TITAN_PANEL_SLASH_RESET_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset: |cffffffffResets "..TITAN_PANEL.." to default values/position.";
+L["TITAN_PANEL_SLASH_RESET_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipfont: |cffffffffResets "..TITAN_PANEL.." tooltip font scale to default.";
+L["TITAN_PANEL_SLASH_RESET_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset tipalpha: |cffffffffResets "..TITAN_PANEL.." tooltip transparency to default.";
+L["TITAN_PANEL_SLASH_RESET_4"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset panelscale: |cffffffffResets "..TITAN_PANEL.." scale to default.";
+L["TITAN_PANEL_SLASH_RESET_5"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."reset spacing: |cffffffffResets "..TITAN_PANEL.." button spacing to default.";
+L["TITAN_PANEL_SLASH_GUI_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {gui control/trans/skin}";
+L["TITAN_PANEL_SLASH_GUI_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui control: |cffffffffOpens the "..TITAN_PANEL.." control GUI.";
+L["TITAN_PANEL_SLASH_GUI_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui trans: |cffffffffOpens the Transparency control GUI.";
+L["TITAN_PANEL_SLASH_GUI_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."gui skin: |cffffffffOpens the Skin control GUI.";
+L["TITAN_PANEL_SLASH_PROFILE_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {profile use <profile>}";
+L["TITAN_PANEL_SLASH_PROFILE_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."profile use <name> <server>: |cffffffffSets the profile to the requested saved profile.";
+L["TITAN_PANEL_SLASH_PROFILE_2"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<name>: |cffffffffcan be either the character name or the custom profile name."
+L["TITAN_PANEL_SLASH_PROFILE_3"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<server>: |cffffffffcan be either the server name or 'TitanCustomProfile'."
+L["TITAN_PANEL_SLASH_SILENT_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {silent}";
+L["TITAN_PANEL_SLASH_SILENT_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."silent: |cffffffffToggles "..TITAN_PANEL.." to load silently.";
+L["TITAN_PANEL_SLASH_ORDERHALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {orderhall}";
+L["TITAN_PANEL_SLASH_ORDERHALL_1"] = LIGHTYELLOW_FONT_COLOR_CODE.."orderhall: |cffffffffToggles "..TITAN_PANEL.." to hide Order Hall Command Bar.";
+L["TITAN_PANEL_SLASH_HELP_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan {help | help <topic>}";
+L["TITAN_PANEL_SLASH_HELP_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: reset/gui/profile/silent/orderhall/help ";
+L["TITAN_PANEL_SLASH_ALL_0"] = LIGHTYELLOW_FONT_COLOR_CODE.."Usage: |cffffffff/titan <topic>";
+L["TITAN_PANEL_SLASH_ALL_1"] = " - "..LIGHTYELLOW_FONT_COLOR_CODE.."<topic>: |cffffffffreset/gui/profile/silent/orderhall/help ";
+
+-- slash command responses
+L["TITAN_PANEL_SLASH_RESP1"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." tooltip font scale has been reset.";
+L["TITAN_PANEL_SLASH_RESP2"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." tooltip transparency has been reset.";
+L["TITAN_PANEL_SLASH_RESP3"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." scale has been reset.";
+L["TITAN_PANEL_SLASH_RESP4"] = LIGHTYELLOW_FONT_COLOR_CODE..TITAN_PANEL.." button spacing has been reset.";
+
+-- global profile locale
+L["TITAN_PANEL_GLOBAL"] = "Global";
+L["TITAN_PANEL_GLOBAL_PROFILE"] = "Global Profile";
+L["TITAN_PANEL_GLOBAL_USE"] = "Use Global Profile";
+L["TITAN_PANEL_GLOBAL_USE_AS"] = "Use as Global Profile";
+L["TITAN_PANEL_GLOBAL_USE_DESC"] = "Use a global profile for all characters";
+L["TITAN_PANEL_GLOBAL_RESET_PART"] = "resetting options";
+L["TITAN_PANEL_GLOBAL_ERR_1"] = "You may not load a profile when a global profile is in use";
+
+-- general panel locale
+L["TITAN_PANEL_VERSION_INFO"] = "|cffffd700 by the |cffff8c00"..TITAN_PANEL.." Development Team";
+L["TITAN_PANEL_MENU_TITLE"] = TITAN_PANEL;
+L["TITAN_PANEL_MENU_HIDE"] = "Hide";
+L["TITAN_PANEL_MENU_IN_COMBAT_LOCKDOWN"] = "(In Combat)";
+L["TITAN_PANEL_MENU_RELOADUI"] = "(Reload UI)";
+L["TITAN_PANEL_MENU_SHOW_COLORED_TEXT"] = "Show Colored Text";
+L["TITAN_PANEL_MENU_SHOW_ICON"] = "Show Icon";
+L["TITAN_PANEL_MENU_SHOW_LABEL_TEXT"] = "Show Label Text";
+L["TITAN_PANEL_MENU_AUTOHIDE"] = "Auto-Hide";
+L["TITAN_PANEL_MENU_CENTER_TEXT"] = "Center Text";
+L["TITAN_PANEL_MENU_DISPLAY_BAR"] = "Show Bar";
+L["TITAN_PANEL_MENU_DISABLE_PUSH"] = "Disable Screen Adjust";
+L["TITAN_PANEL_MENU_DISABLE_MINIMAP_PUSH"] = "Disable Minimap Adjust";
+L["TITAN_PANEL_MENU_DISABLE_LOGS"] = "Automatic Log Adjust";
+L["TITAN_PANEL_MENU_DISABLE_BAGS"] = "Automatic Bag Adjust";
+L["TITAN_PANEL_MENU_DISABLE_TICKET"] = "Automatic Ticket Frame Adjust";
+L["TITAN_PANEL_MENU_PROFILES"] = "Profiles";
+L["TITAN_PANEL_MENU_PROFILE"] = "Profile ";
+L["TITAN_PANEL_MENU_PROFILE_CUSTOM"] = "Custom";
+L["TITAN_PANEL_MENU_PROFILE_DELETED"] = " has been deleted.";
+L["TITAN_PANEL_MENU_PROFILE_SERVERS"] = "Server";
+L["TITAN_PANEL_MENU_PROFILE_CHARS"] = "Character";
+L["TITAN_PANEL_MENU_PROFILE_RELOADUI"] = "Your UI will now reload upon pushing 'Okay' to allow saving of your custom profile.";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_CUSTOM_TITLE"] = "Enter a name for your custom profile:\n(20 chars max, no spaces allowed, case sensitive)";
+L["TITAN_PANEL_MENU_PROFILE_SAVE_PENDING"] = "Current settings are to be saved under profile name: ";
+L["TITAN_PANEL_MENU_PROFILE_ALREADY_EXISTS"] = "The profile name entered already exists. Are you sure you want to overwrite it ? Push 'Accept' if yes, otherwise push 'Cancel' or the 'Escape' key.";
+L["TITAN_PANEL_MENU_MANAGE_SETTINGS"] = "Manage";
+L["TITAN_PANEL_MENU_LOAD_SETTINGS"] = "Load";
+L["TITAN_PANEL_MENU_DELETE_SETTINGS"] = "Delete";
+L["TITAN_PANEL_MENU_SAVE_SETTINGS"] = "Save";
+L["TITAN_PANEL_MENU_CONFIGURATION"] = "Configuration";
+L["TITAN_PANEL_OPTIONS"] = "Options";
+L["TITAN_PANEL_MENU_TOP"] = "Top"
+L["TITAN_PANEL_MENU_TOP2"] = "Top 2"
+L["TITAN_PANEL_MENU_BOTTOM"] = "Bottom"
+L["TITAN_PANEL_MENU_BOTTOM2"] = "Bottom 2"
+L["TITAN_PANEL_MENU_OPTIONS"] = TITAN_PANEL.." Tooltips and Frames";
+L["TITAN_PANEL_MENU_OPTIONS_SHORT"] = "Tooltips and Frames";
+L["TITAN_PANEL_MENU_TOP_BARS"] = "Top Bars"
+L["TITAN_PANEL_MENU_BOTTOM_BARS"] = "Bottom Bars"
+L["TITAN_PANEL_MENU_OPTIONS_BARS"] = "Bars"
+L["TITAN_PANEL_MENU_OPTIONS_MAIN_BARS"] = TITAN_PANEL.." Top Bars";
+L["TITAN_PANEL_MENU_OPTIONS_AUX_BARS"] = TITAN_PANEL.." Bottom Bars";
+L["TITAN_PANEL_MENU_OPTIONS_TOOLTIPS"] = "Tooltips";
+L["TITAN_PANEL_MENU_OPTIONS_FRAMES"] = "Frames";
+L["TITAN_PANEL_MENU_PLUGINS"] = "Plugins";
+L["TITAN_PANEL_MENU_LOCK_BUTTONS"] = "Lock Buttons";
+L["TITAN_PANEL_MENU_VERSION_SHOWN"] = "Show Plugin Versions";
+L["TITAN_PANEL_MENU_LDB_SIDE"] = "Right-Side Plugin";
+L["TITAN_PANEL_MENU_LDB_FORCE_LAUNCHER"] = "Force LDB Launchers to Right-Side";
+L["TITAN_PANEL_MENU_CATEGORIES"] = {"Built-ins","General","Combat","Information","Interface","Profession"}
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN"] = "Show Tooltips";
+L["TITAN_PANEL_MENU_TOOLTIPS_SHOWN_IN_COMBAT"] = "Hide Tooltips in Combat";
+L["TITAN_PANEL_MENU_AUTOHIDE_IN_COMBAT"] = "Lock auto hide bars while in combat";
+L["TITAN_PANEL_MENU_RESET"] = "Reset "..TITAN_PANEL.." to Default";
+L["TITAN_PANEL_MENU_TEXTURE_SETTINGS"] = "Skins";
+L["TITAN_PANEL_MENU_LSM_FONTS"] = "Panel Font"
+L["TITAN_PANEL_MENU_ENABLED"] = "Enabled";
+L["TITAN_PANEL_MENU_DISABLED"] = "Disabled";
+L["TITAN_PANEL_SHIFT_LEFT"] = "Shift Left";
+L["TITAN_PANEL_SHIFT_RIGHT"] = "Shift Right";
+L["TITAN_PANEL_MENU_SHOW_PLUGIN_TEXT"] = "Show Plugin Text";
+L["TITAN_PANEL_MENU_BAR_ALWAYS"] = "Always On";
+L["TITAN_PANEL_MENU_POSITION"] = "Position";
+L["TITAN_PANEL_MENU_BAR"] = "Bar";
+L["TITAN_PANEL_MENU_DISPLAY_ON_BAR"] = "Choose which bar the plugin is displayed";
+L["TITAN_PANEL_MENU_SHOW"] = "Show Plugin";
+L["TITAN_PANEL_MENU_PLUGIN_RESET"] = "Refresh Plugins";
+L["TITAN_PANEL_MENU_PLUGIN_RESET_DESC"] = "Refresh Plugin Text and Position";
+L["TITAN_PANEL_MENU_SILENT_LOAD"] = "Silent Load";
+L["TITAN_PANEL_MENU_HIDE_ORDERHALL"] = "Hide Order Hall Command Bar";
+
+-- localization strings for AceConfigDialog-3.0
+L["TITAN_ABOUT_VERSION"] = "Version";
+L["TITAN_ABOUT_AUTHOR"] = "Author";
+L["TITAN_ABOUT_CREDITS"] = "Credits";
+L["TITAN_ABOUT_CATEGORY"] = "Category";
+L["TITAN_ABOUT_EMAIL"] = "E-mail";
+L["TITAN_ABOUT_WEB"] = "Website";
+L["TITAN_ABOUT_LICENSE"] = "License";
+L["TITAN_PANEL_CONFIG_MAIN_LABEL"] = "Information display bar addon. Allows users to add data feed or launcher plugins on a control panel placed on the top and/or bottom of the screen.";
+L["TITAN_TRANS_MENU_TEXT"] = TITAN_PANEL.." Transparency";
+L["TITAN_TRANS_MENU_TEXT_SHORT"] = "Transparency";
+L["TITAN_TRANS_MENU_DESC"] = "Adjust transparency for the "..TITAN_PANEL.." bars and tooltip.";
+L["TITAN_TRANS_MAIN_CONTROL_TITLE"] = "Main Bar";
+L["TITAN_TRANS_AUX_CONTROL_TITLE"] = "Auxiliary Bar";
+L["TITAN_TRANS_CONTROL_TITLE_TOOLTIP"] = "Tooltip";
+L["TITAN_TRANS_TOOLTIP_DESC"] = "Sets transparency for the tooltip of the various plugins.";
+L["TITAN_UISCALE_MENU_TEXT"] = TITAN_PANEL.." Scale and Font";
+L["TITAN_UISCALE_MENU_TEXT_SHORT"] = "Scale and Font";
+L["TITAN_UISCALE_CONTROL_TITLE_UI"] = "UI Scale";
+L["TITAN_UISCALE_CONTROL_TITLE_PANEL"] = TITAN_PANEL.." Scale";
+L["TITAN_UISCALE_CONTROL_TITLE_BUTTON"] = "Button Spacing";
+L["TITAN_UISCALE_CONTROL_TITLE_ICON"] = "Icon Spacing";
+L["TITAN_UISCALE_CONTROL_TOOLTIP_TOOLTIPFONT"] = "Tooltip Font Scale";
+L["TITAN_UISCALE_TOOLTIP_DISABLE_TEXT"] = "Disable Tooltip Font Scale";
+L["TITAN_UISCALE_MENU_DESC"] = "Controls various aspects of the UI and "..TITAN_PANEL..".";
+L["TITAN_UISCALE_SLIDER_DESC"] = "Sets the scale of your entire UI.";
+L["TITAN_UISCALE_PANEL_SLIDER_DESC"] = "Sets the scale for the various "..TITAN_PANEL.." buttons and icons.";
+L["TITAN_UISCALE_BUTTON_SLIDER_DESC"] = "Adjusts the space between left-side plugins.";
+L["TITAN_UISCALE_ICON_SLIDER_DESC"] = "Adjusts the space between right-side plugins.";
+L["TITAN_UISCALE_TOOLTIP_SLIDER_DESC"] = "Adjusts the scale for the tooltip of the various plugins.";
+L["TITAN_UISCALE_DISABLE_TOOLTIP_DESC"] = "Disables "..TITAN_PANEL.." Tooltip Font Scale Control.";
+
+L["TITAN_SKINS_TITLE"] = TITAN_PANEL.." Skins";
+L["TITAN_SKINS_OPTIONS_CUSTOM"] = "Skins - Custom";
+L["TITAN_SKINS_TITLE_CUSTOM"] = TITAN_PANEL.." Custom Skins";
+L["TITAN_SKINS_MAIN_DESC"] = "All custom skins are assumed to be in: \n"
+			.."..\\AddOns\\Titan\\Artwork\\Custom\\<Skin Folder>\\ ".."\n"
+			.."\n"..TITAN_PANEL.." and custom skins are stored under the Custom folder."
+L["TITAN_SKINS_LIST_TITLE"] = "Skin List";
+L["TITAN_SKINS_SET_DESC"] = "Select a skin for the "..TITAN_PANEL.." bars.";
+L["TITAN_SKINS_SET_HEADER"] = "Set "..TITAN_PANEL.." Skin";
+L["TITAN_SKINS_RESET_HEADER"] = "Reset "..TITAN_PANEL.." Skins";
+L["TITAN_SKINS_NEW_HEADER"] = "Add New Skin";
+L["TITAN_SKINS_NAME_TITLE"] = "Skin Name"
+L["TITAN_SKINS_NAME_DESC"] = "Enter a name for your new skin. It will be used in the skin dropdown lists.";
+L["TITAN_SKINS_PATH_TITLE"] = "<Skin Folder>"
+L["TITAN_SKINS_PATH_DESC"] = "<Skin Folder> under the "..TITAN_PANEL.." install. See the example above."
+L["TITAN_SKINS_ADD_HEADER"] = "Add Skin";
+L["TITAN_SKINS_ADD_DESC"] = "Adds a new skin to the list of available skins for "..TITAN_PANEL..".";
+L["TITAN_SKINS_REMOVE_HEADER"] = "Remove Skin";
+L["TITAN_SKINS_REMOVE_DESC"] = "Select a custom skin to remove."
+L["TITAN_SKINS_REMOVE_BUTTON"] = "Remove";
+L["TITAN_SKINS_REMOVE_BUTTON_DESC"] = "Removes the selected custom skin.";
+L["TITAN_SKINS_REMOVE_NOTES"] = "You are responsible for removing any unwanted custom skins "
+	.."from the "..TITAN_PANEL.." install folder. Addons can not add or remove files."
+L["TITAN_SKINS_RESET_DEFAULTS_TITLE"] = "Reset to Defaults";
+L["TITAN_SKINS_RESET_DEFAULTS_DESC"] = "Resets the skin list to the default "..TITAN_PANEL.." skins.";
+L["TITAN_PANEL_MENU_LSM_FONTS_DESC"] = "Select the font type for the various plugins on the "..TITAN_PANEL.." Bars.";
+L["TITAN_PANEL_MENU_FONT_SIZE"] = "Font Size";
+L["TITAN_PANEL_MENU_FONT_SIZE_DESC"] = "Sets the size for the "..TITAN_PANEL.." font.";
+L["TITAN_PANEL_MENU_FRAME_STRATA"] = ""..TITAN_PANEL.." Frame Strata";
+L["TITAN_PANEL_MENU_FRAME_STRATA_DESC"] = "Adjusts the frame strata for the "..TITAN_PANEL.." Bar(s).";
+-- /end localization strings for AceConfigDialog-3.0
+
+L["TITAN_PANEL_MENU_ADV"] = "Advanced";
+L["TITAN_PANEL_MENU_ADV_DESC"] = "Change Timers only if you experience issues with frames not adjusting.".."\n";
+L["TITAN_PANEL_MENU_ADV_PEW"] = "Entering World";
+L["TITAN_PANEL_MENU_ADV_PEW_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving world or an instance.";
+L["TITAN_PANEL_MENU_ADV_VEHICLE"] = "Vehicle";
+L["TITAN_PANEL_MENU_ADV_VEHICLE_DESC"] = "Change value (usually increase) if frames do not adjust when entering / leaving vehicle.";
+
+L["TITAN_AUTOHIDE_TOOLTIP"] = "Toggles "..TITAN_PANEL.." auto-hide on/off feature";
+
+L["TITAN_BAG_FORMAT"] = "%d/%d";
+L["TITAN_BAG_BUTTON_LABEL"] = "Bags: ";
+L["TITAN_BAG_TOOLTIP"] = "Bags Info";
+L["TITAN_BAG_TOOLTIP_HINTS"] = "Hint: Left-click to open all bags.";
+L["TITAN_BAG_MENU_TEXT"] = "Bag";
+L["TITAN_BAG_USED_SLOTS"] = "Used Slots";
+L["TITAN_BAG_FREE_SLOTS"] = "Free Slots";
+L["TITAN_BAG_BACKPACK"] = "Backpack";
+L["TITAN_BAG_MENU_SHOW_USED_SLOTS"] = "Show Used Slots";
+L["TITAN_BAG_MENU_SHOW_AVAILABLE_SLOTS"] = "Show Available Slots";
+L["TITAN_BAG_MENU_SHOW_DETAILED"] = "Show Detailed Tooltip";
+L["TITAN_BAG_MENU_IGNORE_SLOTS"] = "Ignore Containers";
+L["TITAN_BAG_MENU_IGNORE_PROF_BAGS_SLOTS"] = "Ignore Profession Bags";
+
+L["TITAN_BAG_PROF_BAG_ENCHANTING"] = {
+"Enchanted Mageweave Pouch", "Enchanted Runecloth Bag", "Enchanter's Satchel", "Big Bag of Enchantment", "Spellfire Bag",
+"Mysterious Bag", "Otherworldly Bag", "\"Carriage - Exclusive\" Enchanting Evening Purse"};
+L["TITAN_BAG_PROF_BAG_ENGINEERING"] = {
+"Heavy Toolbox", "Fel Iron Toolbox", "Titanium Toolbox", "Khorium Toolbox", "Elementium Toolbox", "\"Carriage - Maddy\" High Tech Bag"};
+L["TITAN_BAG_PROF_BAG_HERBALISM"] = {
+"Herb Pouch", "Cenarion Herb Bag", "Satchel of Cenarius", "Mycah's Botanical Bag", "Emerald Bag", "Hyjal Expedition Bag",
+"\"Carriage - Going Green\" Herb Tote Bag"};
+L["TITAN_BAG_PROF_BAG_INSCRIPTION"] = {
+"Scribe's Satchel", "Pack of Endless Pockets", "\"Carriage - Xandera\" Student's Satchel", "Burnished Inscription Bag"};
+L["TITAN_BAG_PROF_BAG_JEWELCRAFTING"] = {
+"Gem Pouch", "Bag of Jewels", "\"Carriage - Exclusive\" Gem Studded Clutch"};
+L["TITAN_BAG_PROF_BAG_LEATHERWORKING"] = {
+"Leatherworker's Satchel", "Bag of Many Hides", "Trapper's Traveling Pack", "\"Carriage - Meeya\" Leather Bag", "Burnished Leather Bag"};
+L["TITAN_BAG_PROF_BAG_MINING"] = {
+"Mining Sack", "Reinforced Mining Bag", "Mammoth Mining Bag", "\"Carriage - Christina\" Precious Metal Bag", "Burnished Mining Bag"};
+L["TITAN_BAG_PROF_BAG_FISHING"] = {"Lure Master Tackle Box"};
+L["TITAN_BAG_PROF_BAG_COOKING"] = {"Portable Refrigerator", "Advanced Refrigeration Unit"};
+
+L["TITAN_CLOCK_TOOLTIP"] = "Clock";
+L["TITAN_CLOCK_TOOLTIP_VALUE"] = "Server Offset Hour Value: ";
+L["TITAN_CLOCK_TOOLTIP_LOCAL_TIME"] = "Local Time: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_TIME"] = "Server Time: ";
+L["TITAN_CLOCK_TOOLTIP_SERVER_ADJUSTED_TIME"] = "Adjusted Server Time: ";
+L["TITAN_CLOCK_TOOLTIP_HINT1"] = "Hint: Left-click to adjust the offset hour"
+L["TITAN_CLOCK_TOOLTIP_HINT2"] = "(server time only) and the 12/24H time format.";
+L["TITAN_CLOCK_TOOLTIP_HINT3"] = "Shift Left-Click to toggle the Calendar on/off.";
+L["TITAN_CLOCK_CONTROL_TOOLTIP"] = "Server Hour Offset: ";
+L["TITAN_CLOCK_CONTROL_TITLE"] = "Offset";
+L["TITAN_CLOCK_CONTROL_HIGH"] = "+12";
+L["TITAN_CLOCK_CONTROL_LOW"] = "-12";
+L["TITAN_CLOCK_CHECKBUTTON"] = "24H Fmt";
+L["TITAN_CLOCK_CHECKBUTTON_TOOLTIP"] = "Toggles the time display between 12-hour and 24-hour format";
+L["TITAN_CLOCK_MENU_TEXT"] = "Clock";
+L["TITAN_CLOCK_MENU_LOCAL_TIME"] = "Show Local Time (L)";
+L["TITAN_CLOCK_MENU_SERVER_TIME"] = "Show Server Time (S)";
+L["TITAN_CLOCK_MENU_SERVER_ADJUSTED_TIME"] = "Show Server Adjusted Time (A)";
+L["TITAN_CLOCK_MENU_DISPLAY_ON_RIGHT_SIDE"] = "Display on Right Side";
+L["TITAN_CLOCK_MENU_HIDE_GAMETIME"] = "Hide Time/Calendar Button";
+L["TITAN_CLOCK_MENU_HIDE_MAPTIME"] = "Hide Time Button";
+L["TITAN_CLOCK_MENU_HIDE_CALENDAR"] = "Hide Calendar Button";
+
+L["TITAN_LOCATION_FORMAT"] = "(%.d, %.d)";
+L["TITAN_LOCATION_FORMAT2"] = "(%.1f, %.1f)";
+L["TITAN_LOCATION_FORMAT3"] = "(%.2f, %.2f)";
+L["TITAN_LOCATION_FORMAT_LABEL"] = "(xx , yy)";
+L["TITAN_LOCATION_FORMAT2_LABEL"] = "(xx.x , yy.y)";
+L["TITAN_LOCATION_FORMAT3_LABEL"] = "(xx.xx , yy.yy)";
+L["TITAN_LOCATION_FORMAT_COORD_LABEL"] = "Coordinate Format";
+L["TITAN_LOCATION_BUTTON_LABEL"] = "Loc: ";
+L["TITAN_LOCATION_TOOLTIP"] = "Location Info";
+L["TITAN_LOCATION_TOOLTIP_HINTS_1"] = "Hint: Shift + left-click to add location";
+L["TITAN_LOCATION_TOOLTIP_HINTS_2"] = "info to the chat message.";
+L["TITAN_LOCATION_TOOLTIP_ZONE"] = "Zone: ";
+L["TITAN_LOCATION_TOOLTIP_SUBZONE"] = "SubZone: ";
+L["TITAN_LOCATION_TOOLTIP_PVPINFO"] = "PVP Info: ";
+L["TITAN_LOCATION_TOOLTIP_HOMELOCATION"] = "Home Location";
+L["TITAN_LOCATION_TOOLTIP_INN"] = "Inn: ";
+L["TITAN_LOCATION_MENU_TEXT"] = "Location";
+L["TITAN_LOCATION_MENU_SHOW_ZONE_ON_PANEL_TEXT"] = "Show Zone Text";
+L["TITAN_LOCATION_MENU_SHOW_COORDS_ON_MAP_TEXT"] = "Show Coordinates on World Map";
+L["TITAN_LOCATION_MAP_CURSOR_COORDS_TEXT"] = "Cursor: %s";
+L["TITAN_LOCATION_MAP_PLAYER_COORDS_TEXT"] = "Player: %s";
+L["TITAN_LOCATION_NO_COORDS"] = "No Coords";
+L["TITAN_LOCATION_MENU_SHOW_LOC_ON_MINIMAP_TEXT"] = "Show Location Name Above Minimap";
+L["TITAN_LOCATION_MENU_UPDATE_WORLD_MAP"] = "Update World Map When Zone Changes";
+
+L["TITAN_FPS_FORMAT"] = "%.1f";
+L["TITAN_FPS_BUTTON_LABEL"] = "FPS: ";
+L["TITAN_FPS_MENU_TEXT"] = "FPS";
+L["TITAN_FPS_TOOLTIP_CURRENT_FPS"] = "Current FPS: ";
+L["TITAN_FPS_TOOLTIP_AVG_FPS"] = "Average FPS: ";
+L["TITAN_FPS_TOOLTIP_MIN_FPS"] = "Minimum FPS: ";
+L["TITAN_FPS_TOOLTIP_MAX_FPS"] = "Maximum FPS: ";
+L["TITAN_FPS_TOOLTIP"] = "Frames Per Second";
+
+L["TITAN_LATENCY_FORMAT"] = "%d".."ms";
+L["TITAN_LATENCY_BANDWIDTH_FORMAT"] = "%.3f ".."KB/s";
+L["TITAN_LATENCY_BUTTON_LABEL"] = "Latency: ";
+L["TITAN_LATENCY_TOOLTIP"] = "Network Status";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_HOME"] = "Realm Latency (home): ";
+L["TITAN_LATENCY_TOOLTIP_LATENCY_WORLD"] = "Game Latency (world): ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_IN"] = "Bandwidth In: ";
+L["TITAN_LATENCY_TOOLTIP_BANDWIDTH_OUT"] = "Bandwidth Out: ";
+L["TITAN_LATENCY_MENU_TEXT"] = "Latency";
+
+L["TITAN_LOOTTYPE_BUTTON_LABEL"] = "Loot: ";
+L["TITAN_LOOTTYPE_FREE_FOR_ALL"] = "Free For All";
+L["TITAN_LOOTTYPE_ROUND_ROBIN"] = "Round Robin";
+L["TITAN_LOOTTYPE_MASTER_LOOTER"] = "Master Looter";
+L["TITAN_LOOTTYPE_GROUP_LOOT"] = "Group Loot";
+L["TITAN_LOOTTYPE_NEED_BEFORE_GREED"] = "Need Before Greed";
+L["TITAN_LOOTTYPE_PERSONAL"] = "Personal";
+L["TITAN_LOOTTYPE_TOOLTIP"] = "Loot Type Info";
+L["TITAN_LOOTTYPE_MENU_TEXT"] = "Loot Type";
+L["TITAN_LOOTTYPE_RANDOM_ROLL_LABEL"] = "Random Roll";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT1"] = "Hint: Left-click for random roll.";
+L["TITAN_LOOTTYPE_TOOLTIP_HINT2"] = "Select roll type from right-click menu.";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL"] = "Dungeon Difficulty";
+L["TITAN_LOOTTYPE_DUNGEONDIFF_LABEL2"] = "Raid Difficulty";
+L["TITAN_LOOTTYPE_SHOWDUNGEONDIFF_LABEL"] = "Show Dungeon/Raid Difficulty";
+L["TITAN_LOOTTYPE_SETDUNGEONDIFF_LABEL"] = "Set Dungeon Difficulty";
+L["TITAN_LOOTTYPE_SETRAIDDIFF_LABEL"] = "Set Raid Difficulty";
+L["TITAN_LOOTTYPE_AUTODIFF_LABEL"] = "Auto (Group Based)";
+
+L["TITAN_MEMORY_FORMAT"] = "%.3f".."MB";
+L["TITAN_MEMORY_FORMAT_KB"] = "%d".."KB";
+L["TITAN_MEMORY_RATE_FORMAT"] = "%.3f".."KB/s";
+L["TITAN_MEMORY_BUTTON_LABEL"] = "Memory: ";
+L["TITAN_MEMORY_TOOLTIP"] = "Memory Usage";
+L["TITAN_MEMORY_TOOLTIP_CURRENT_MEMORY"] = "Current: ";
+L["TITAN_MEMORY_TOOLTIP_INITIAL_MEMORY"] = "Initial: ";
+L["TITAN_MEMORY_TOOLTIP_INCREASING_RATE"] = "Increasing Rate: ";
+L["TITAN_MEMORY_KBMB_LABEL"] = "KB/MB";
+
+L["TITAN_PERFORMANCE_TOOLTIP"] = "Performance Info";
+L["TITAN_PERFORMANCE_MENU_TEXT"] = "Performance";
+L["TITAN_PERFORMANCE_ADDONS"] = "Addon Usage";
+L["TITAN_PERFORMANCE_ADDON_MEM_USAGE_LABEL"] = "Addon Memory Usage";
+L["TITAN_PERFORMANCE_ADDON_MEM_FORMAT_LABEL"] = "Addon Memory Format";
+L["TITAN_PERFORMANCE_ADDON_CPU_USAGE_LABEL"] = "Addon CPU Usage";
+L["TITAN_PERFORMANCE_ADDON_NAME_LABEL"] = "Name:";
+L["TITAN_PERFORMANCE_ADDON_USAGE_LABEL"] = "Usage";
+L["TITAN_PERFORMANCE_ADDON_RATE_LABEL"] = "Rate";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_MEM_USAGE_LABEL"] = "Total Addon Memory:";
+L["TITAN_PERFORMANCE_ADDON_TOTAL_CPU_USAGE_LABEL"] = "Total CPU Time:";
+L["TITAN_PERFORMANCE_MENU_SHOW_FPS"] = "Show FPS";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY"] = "Show Realm Latency";
+L["TITAN_PERFORMANCE_MENU_SHOW_LATENCY_WORLD"] = "Show Game Latency";
+L["TITAN_PERFORMANCE_MENU_SHOW_MEMORY"] = "Show Memory";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDONS"] = "Show Addon Memory Usage";
+L["TITAN_PERFORMANCE_MENU_SHOW_ADDON_RATE"] = "Show Addon Usage Rate";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL"] = "CPU Profiling Mode";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_ON"] = "Enable CPU Profiling Mode ";
+L["TITAN_PERFORMANCE_MENU_CPUPROF_LABEL_OFF"] = "Disable CPU Profiling Mode ";
+L["TITAN_PERFORMANCE_CONTROL_TOOLTIP"] = "Monitored Addons: ";
+L["TITAN_PERFORMANCE_CONTROL_TITLE"] = "Monitored Addons";
+L["TITAN_PERFORMANCE_CONTROL_HIGH"] = "40";
+L["TITAN_PERFORMANCE_CONTROL_LOW"] = "1";
+L["TITAN_PERFORMANCE_TOOLTIP_HINT"] = "Hint: Left-click to force a garbage collection.";
+
+L["TITAN_XP_FORMAT"] = "%s";
+L["TITAN_XP_PERCENT_FORMAT"] = "(%.1f%%)";
+L["TITAN_XP_BUTTON_LABEL_XPHR_LEVEL"] = "XP/hr This Level: ";
+L["TITAN_XP_BUTTON_LABEL_XPHR_SESSION"] = "XP/hr This Session: ";
+L["TITAN_XP_BUTTON_LABEL_TOLEVEL_TIME_LEVEL"] = "Time To Level: ";
+L["TITAN_XP_LEVEL_COMPLETE"] = "Level Complete: ";
+L["TITAN_XP_TOTAL_RESTED"] = "Rested: ";
+L["TITAN_XP_XPTOLEVELUP"] = "XP To Level: ";
+L["TITAN_XP_TOOLTIP"] = "XP Info";
+L["TITAN_XP_TOOLTIP_TOTAL_TIME"] = "Total Time Played: ";
+L["TITAN_XP_TOOLTIP_LEVEL_TIME"] = "Time Played This Level: ";
+L["TITAN_XP_TOOLTIP_SESSION_TIME"] = "Time Played This Session: ";
+L["TITAN_XP_TOOLTIP_TOTAL_XP"] = "Total XP Required This Level: ";
+L["TITAN_XP_TOOLTIP_LEVEL_XP"] = "XP Gained This Level: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_XP"] = "XP Needed To Level: ";
+L["TITAN_XP_TOOLTIP_SESSION_XP"] = "XP Gained This Session: ";
+L["TITAN_XP_TOOLTIP_XPHR_LEVEL"] = "XP/HR This Level: ";
+L["TITAN_XP_TOOLTIP_XPHR_SESSION"] = "XP/HR This Session: ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_LEVEL"] = "Time To Level (Level Rate): ";
+L["TITAN_XP_TOOLTIP_TOLEVEL_SESSION"] = "Time To Level (Session Rate): ";
+L["TITAN_XP_MENU_TEXT"] = "XP";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_LEVEL"] = "Show XP/HR This Level";
+L["TITAN_XP_MENU_SHOW_XPHR_THIS_SESSION"] = "Show XP/HR This Session";
+L["TITAN_XP_MENU_SHOW_RESTED_TOLEVELUP"] = "Show Multi-Info View";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TITLE"] = "Button";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_RESTED"] = "Show Rested XP";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_TOLEVELUP"] = "Show XP To Level";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_KILLS"] = "Show Estimated Kills To Level";
+L["TITAN_XP_MENU_RESET_SESSION"] = "Reset Session";
+L["TITAN_XP_MENU_REFRESH_PLAYED"] = "Refresh Timers";
+L["TITAN_XP_UPDATE_PENDING"] = "Updating...";
+L["TITAN_XP_KILLS_LABEL"] = "Kills To Level (at %s XP gained last): ";
+L["TITAN_XP_KILLS_LABEL_SHORT"] = "Est. Kills: ";
+L["TITAN_XP_BUTTON_LABEL_SESSION_TIME"] = "Session Time: ";
+L["TITAN_XP_MENU_SHOW_SESSION_TIME"] = "Show Session Time";
+L["TITAN_XP_GAIN_PATTERN"] = "(.*) dies, you gain (%d+) experience.";
+L["TITAN_XP_XPGAINS_LABEL_SHORT"] = "Est. Gains: ";
+L["TITAN_XP_XPGAINS_LABEL"] = "XP Gains To Level (at %s XP gained last): ";
+L["TITAN_XP_MENU_SIMPLE_BUTTON_XPGAIN"] = "Show Estimated XP Gains To Level";
+
+--Titan Repair
+L["REPAIR_LOCALE"] = {
+	menu = "Repair",
+	tooltip = "Repair Info",
+	button = "Durability: ",
+	normal = "Repair Cost (Normal): ",
+	friendly = "Repair Cost (Friendly): ",
+	honored = "Repair Cost (Honored): ",
+	revered = "Repair Cost (Revered): ",
+	exalted = "Repair Cost (Exalted): ",
+	buttonNormal = "Show Normal",
+	buttonFriendly = "Show Friendly (5%)",
+	buttonHonored = "Show Honored (10%)",
+	buttonRevered = "Show Revered (15%)",
+	buttonExalted = "Show Exalted (20%)",
+	percentage = "Show as Percentage",
+	itemnames = "Show Item Names",
+	mostdamaged = "Show Most Damaged",
+	showdurabilityframe = "Show Durability Frame",
+	undamaged = "Show Undamaged Items",
+	discount = "Discount",
+	nothing = "Nothing Damaged",
+	confirmation = "Do you want to repair all items ?",
+	badmerchant = "This merchant cannot repair. Displaying normal repair costs instead.",
+	popup = "Show Repair Popup",
+	showinventory = "Calculate Inventory Damage",
+	WholeScanInProgress = "Updating...",
+	AutoReplabel = "Auto-Repair",
+	AutoRepitemlabel = "Auto Repair All Items",
+	ShowRepairCost = "Show Repair Cost",
+	ignoreThrown = "Ignore Thrown",
+	ShowItems = "Show Items",
+	ShowDiscounts = "Show Discounts",
+	ShowCosts = "Show Costs",
+	Items = "Items",
+	Discounts = "Discounts",
+	Costs = "Costs",
+	CostTotal = "Total Cost",
+	CostBag = "Bag Cost",
+	CostEquip = "Equip Cost",
+	TooltipOptions = "Tooltip",
+};
+L["TITAN_REPAIR"] = "Titan Repair"
+L["TITAN_REPAIR_GBANK_TOTAL"] = "Guild Bank Funds :"
+L["TITAN_REPAIR_GBANK_WITHDRAW"] = "Guild Bank Withdrawal Allowed :"
+L["TITAN_REPAIR_GBANK_USEFUNDS"] = "Use Guild Bank Funds"
+L["TITAN_REPAIR_GBANK_NOMONEY"] = "Guild Bank can't afford the repair cost, or you can't withdraw that much."
+L["TITAN_REPAIR_GBANK_NORIGHTS"] = "You are either not in a guild or you don't have permission to use the guild bank to repair your items."
+L["TITAN_REPAIR_CANNOT_AFFORD"] = "You cannot afford to repair, at this time."
+L["TITAN_REPAIR_REPORT_COST_MENU"] = "Report Repair Cost to Chat"
+L["TITAN_REPAIR_REPORT_COST_CHAT"] = "Repair cost was "
+
+L["TITAN_GOLD_TOOLTIPTEXT"] = "Total Gold on";
+L["TITAN_GOLD_ITEMNAME"] = "Titan Gold";
+L["TITAN_GOLD_CLEAR_DATA_TEXT"] = "Clear Database";
+L["TITAN_GOLD_RESET_SESS_TEXT"] = "Reset Current Session";
+L["TITAN_GOLD_DB_CLEARED"] = "Titan Gold - Database Cleared.";
+L["TITAN_GOLD_SESSION_RESET"] = "Titan Gold - Session Reset.";
+L["TITAN_GOLD_MENU_TEXT"] = "Gold";
+L["TITAN_GOLD_TOOLTIP"] = "Gold Info";
+L["TITAN_GOLD_TOGGLE_PLAYER_TEXT"] = "Display Player Gold";
+L["TITAN_GOLD_TOGGLE_ALL_TEXT"] = "Display Server Gold";
+L["TITAN_GOLD_SESS_EARNED"] = "Earned This Session";
+L["TITAN_GOLD_PERHOUR_EARNED"] = "Earned Per Hour";
+L["TITAN_GOLD_SESS_LOST"] = "Lost This Session";
+L["TITAN_GOLD_PERHOUR_LOST"] = "Lost Per Hour";
+L["TITAN_GOLD_STATS_TITLE"] = "Session Statistics";
+L["TITAN_GOLD_TTL_GOLD"] = "Total Gold";
+L["TITAN_GOLD_START_GOLD"] = "Starting Gold";
+L["TITAN_GOLD_TOGGLE_SORT_GOLD"] = "Sort Table By Gold";
+L["TITAN_GOLD_TOGGLE_SORT_NAME"] = "Sort Gold By Name";
+L["TITAN_GOLD_TOGGLE_GPH_SHOW"] = "Display Gold Per Hour";
+L["TITAN_GOLD_TOGGLE_GPH_HIDE"] = "Hide Gold Per Hour";
+L["TITAN_GOLD_GOLD"] = "g";
+L["TITAN_GOLD_SILVER"] = "s";
+L["TITAN_GOLD_COPPER"] = "c";
+L["TITAN_GOLD_STATUS_PLAYER_SHOW"] = "Visible";
+L["TITAN_GOLD_STATUS_PLAYER_HIDE"] = "Hidden";
+L["TITAN_GOLD_DELETE_PLAYER"] = "Delete Toon";
+L["TITAN_GOLD_SHOW_PLAYER"] = "Show Toon";
+L["TITAN_GOLD_FACTION_PLAYER_ALLY"] = "Alliance";
+L["TITAN_GOLD_FACTION_PLAYER_HORDE"] = "Horde";
+L["TITAN_GOLD_CLEAR_DATA_WARNING"] = GREEN_FONT_COLOR_CODE.."Warning: "
+..FONT_COLOR_CODE_CLOSE.."This setting will wipe your Titan Gold database. "
+.."If you wish to continue with this operation, push 'Accept', otherwise push 'Cancel' or the 'Escape' key.";
+L["TITAN_GOLD_COIN_NONE"] = "Show No Labels";
+L["TITAN_GOLD_COIN_LABELS"] = "Show Text Labels";
+L["TITAN_GOLD_COIN_ICONS"] = "Show Icon Labels";
+L["TITAN_GOLD_ONLY"] = "Show Gold Only";
+L["TITAN_GOLD_COLORS"] = "Show Gold Colors";
+L["TITAN_GOLD_MERGE"] = "Merge Servers";
+L["TITAN_GOLD_SEPARATE"] = "Separate Servers";
+
+L["TITAN_VOLUME_TOOLTIP"] = "Volume Info";
+L["TITAN_VOLUME_MASTER_TOOLTIP_VALUE"] = "Master Sound Volume: ";
+L["TITAN_VOLUME_SOUND_TOOLTIP_VALUE"] = "Effects Sound Volume: ";
+L["TITAN_VOLUME_AMBIENCE_TOOLTIP_VALUE"] = "Ambience Sound Volume: ";
+L["TITAN_VOLUME_DIALOG_TOOLTIP_VALUE"] = "Dialog Sound Volume: ";
+L["TITAN_VOLUME_MUSIC_TOOLTIP_VALUE"] = "Music Sound Volume: ";
+L["TITAN_VOLUME_MICROPHONE_TOOLTIP_VALUE"] = "Microphone Sound Volume: ";
+L["TITAN_VOLUME_SPEAKER_TOOLTIP_VALUE"] = "Speaker Sound Volume: ";
+L["TITAN_VOLUME_TOOLTIP_HINT1"] = "Hint: Left-click to adjust the"
+L["TITAN_VOLUME_TOOLTIP_HINT2"] = "sound volume.";
+L["TITAN_VOLUME_CONTROL_TOOLTIP"] = "Volume Control: ";
+L["TITAN_VOLUME_CONTROL_TITLE"] = "Volume Control";
+L["TITAN_VOLUME_MASTER_CONTROL_TITLE"] = "Master";
+L["TITAN_VOLUME_SOUND_CONTROL_TITLE"] = "Effects";
+L["TITAN_VOLUME_AMBIENCE_CONTROL_TITLE"] = "Ambience";
+L["TITAN_VOLUME_DIALOG_CONTROL_TITLE"] = "Dialog";
+L["TITAN_VOLUME_MUSIC_CONTROL_TITLE"] = "Music";
+L["TITAN_VOLUME_MICROPHONE_CONTROL_TITLE"] = "Microphone";
+L["TITAN_VOLUME_SPEAKER_CONTROL_TITLE"] = "Speaker";
+L["TITAN_VOLUME_CONTROL_HIGH"] = "High";
+L["TITAN_VOLUME_CONTROL_LOW"] = "Low";
+L["TITAN_VOLUME_MENU_TEXT"] = "Volume Control";
+L["TITAN_VOLUME_MENU_AUDIO_OPTIONS_LABEL"] = "Show Sound/Voice Options" ;
+L["TITAN_VOLUME_MENU_OVERRIDE_BLIZZ_SETTINGS"] = "Override Blizzard Volume Settings";