4 Commits
0.5.0 ... 0.7.2

Author SHA1 Message Date
5b72ad3b78 Version 0.7.2-beta
- minimap button

- graphical move arrows

- crash on matches with 0-width
2020-06-08 01:55:52 +02:00
8e179692ee Version 0.7.1-beta
- user reporting capabilities
- ignore Battle.net contacts
- screenshots

- graphical move arrows
2020-06-07 15:01:42 +02:00
cb2c995a82 Version 0.7.0-beta
- order buttons

- use numeric LogLevel over booleans
- exact case option reversed (again)
- smart case handling if replacement is longer than match

- Deletion of all mappings
2020-06-07 02:42:26 +02:00
a29f6486fe Version 0.6.0
- honour capital/mixed cases for ignore case
- consolidate consecutive matches
- database upgrade capability

- case sensitivity option reversed
- removed obsolete WIM handling

- skip empty mapping
- Deletion of mapping
2020-06-06 14:49:55 +02:00
30 changed files with 1755 additions and 452 deletions

1
.gitignore vendored
View File

@@ -37,6 +37,7 @@ out/
obj/ obj/
# Misc # Misc
.bin/
*.log *.log
*.graphml *.graphml
coverage.db* coverage.db*

View File

@@ -6,7 +6,10 @@ externals:
libs/LibStub: https://repos.wowace.com/wow/libstub/tags/1.0 libs/LibStub: https://repos.wowace.com/wow/libstub/tags/1.0
libs/CallbackHandler-1.0: https://repos.wowace.com/wow/callbackhandler/trunk/CallbackHandler-1.0 libs/CallbackHandler-1.0: https://repos.wowace.com/wow/callbackhandler/trunk/CallbackHandler-1.0
libs/AceAddon-3.0: https://repos.wowace.com/wow/ace3/trunk/AceAddon-3.0 libs/AceAddon-3.0: https://repos.wowace.com/wow/ace3/trunk/AceAddon-3.0
libs/AceConfig-3.0: https://repos.wowace.com/wow/ace3/trunk/AceConfig-3.0
libs/AceConsole-3.0: https://repos.wowace.com/wow/ace3/trunk/AceConsole-3.0 libs/AceConsole-3.0: https://repos.wowace.com/wow/ace3/trunk/AceConsole-3.0
libs/AceDB-3.0: https://repos.wowace.com/wow/ace3/trunk/AceDB-3.0
libs/AceDBOptions-3.0: https://repos.wowace.com/wow/ace3/trunk/AceDBOptions-3.0
libs/AceEvent-3.0: https://repos.wowace.com/wow/ace3/trunk/AceEvent-3.0 libs/AceEvent-3.0: https://repos.wowace.com/wow/ace3/trunk/AceEvent-3.0
libs/AceGUI-3.0: https://repos.wowace.com/wow/ace3/trunk/AceGUI-3.0 libs/AceGUI-3.0: https://repos.wowace.com/wow/ace3/trunk/AceGUI-3.0
libs/AceHook-3.0: https://repos.wowace.com/wow/ace3/trunk/AceHook-3.0 libs/AceHook-3.0: https://repos.wowace.com/wow/ace3/trunk/AceHook-3.0

View File

@@ -3,17 +3,53 @@ All notable changes to this project will be documented in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
## [Unreleased] Version 1.0 - 2020-06-02 ## [Upcoming] Version 0.8.0-beta - 2020-06-08
### Added ### Added
- case sensitivity - handle replacement via slash command
- emote detection
## Version 0.7.2-beta - 2020-06-08
### Added
- minimap button
### Changed
- graphical move arrows
###Fixed
- crash on matches with 0-width
## Version 0.7.1-beta - 2020-06-07
### Added
- user reporting capabilities
- ignore Battle.net contacts
### Changed
- graphical move arrows
## Version 0.7.0-beta - 2020-06-07
### Added
- order buttons
### Changed
- use numeric LogLevel over booleans
- exact case option reversed (again)
- smart case handling if replacement is longer than match
### Fixed
- Deletion of all mappings
## Version 0.6.0 (unreleased) - 2020-06-05
### Added
- honour capital/mixed cases for ignore case
- consolidate consecutive matches - consolidate consecutive matches
- database upgrade capability
### Changed
- case sensitivity option reversed
- removed obsolete WIM handling
### Fixed
- skip empty mapping
- Deletion of mapping
## Version 0.5.0 - 2020-06-01 ## Version 0.5.0 - 2020-06-01
### Added ### Added
- add replacements via options UI - add replacements via options UI
- handle replacement via slash command
## Version 0.4.0 - 2020-05-30 ## Version 0.4.0 (unreleased) - 2020-05-30
### Added ### Added
- restructured files - restructured files
- extract functions and color codes - extract functions and color codes
@@ -23,13 +59,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
### Fixed ### Fixed
- fixed DB storange and debug printing - fixed DB storange and debug printing
## Version 0.2.2 - 2020-05-26 ## Version 0.2.2 (unreleased) - 2020-05-26
### Added ### Added
- added Options UI under Interface Options - added Options UI under Interface Options
- store settings in profiles - store settings in profiles
- added more translations - added more translations
## Version 0.2.1 - 2020-05-25 ## Version 0.2.1 (unreleased) - 2020-05-25
### Added ### Added
- support automatic packaging for curseforge via .pkgmeta - support automatic packaging for curseforge via .pkgmeta
- include project logo - include project logo
@@ -42,6 +78,6 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- handle SendChatMessage ordering if addon Misspelled is also installed - handle SendChatMessage ordering if addon Misspelled is also installed
- break long texts in chunks of 255 length - break long texts in chunks of 255 length
## Version 0.1 - 2020-05-24 ## Version 0.1 (unreleased) - 2020-05-24
### Added ### Added
- bootstrap addon with Ace3 based on [Misspelled](https://www.curseforge.com/wow/addons/misspelled) - bootstrap addon with Ace3 based on [Misspelled](https://www.curseforge.com/wow/addons/misspelled)

View File

@@ -1,17 +1,15 @@
--[[--------------------------------------------------------------------------- --[[---------------------------------------------------------------------------
Grichelde
Grichelde - Text Replacer
Copyright 2020 Teilzeit-Jedi <tj@teilzeit-jedi.de> Copyright 2020 Teilzeit-Jedi <tj@teilzeit-jedi.de>
based on Misspelled developed by Nathan Pieper - nrpieper (@) gmail (dot) com This addon is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
This code freely distributed for your use in any GPL compliant project. You should have received a copy of the GNU General Public License
See conditions in the LICENSE file attached. along with the addon. If not, see <http://www.gnu.org/licenses/gpl-3.0.txt>.
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
-----------------------------------------------------------------------------]] -----------------------------------------------------------------------------]]
@@ -24,9 +22,8 @@ local Grichelde = LibStub("AceAddon-3.0"):NewAddon(AddonTable, AddonName, "AceCo
Grichelde.L = LibStub("AceLocale-3.0"):GetLocale("Grichelde", true) Grichelde.L = LibStub("AceLocale-3.0"):GetLocale("Grichelde", true)
Grichelde.version = GetAddOnMetadata(AddonName, "Version") Grichelde.version = GetAddOnMetadata(AddonName, "Version")
Grichelde.build = GetAddOnMetadata(AddonName, "X-Build") or "Experimental" Grichelde.build = GetAddOnMetadata(AddonName, "X-Build") or "Experimental"
Grichelde.hooks = {}
Grichelde.classic = _G.WOW_PROJECT_ID == _G.WOW_PROJECT_CLASSIC Grichelde.classic = _G.WOW_PROJECT_ID == _G.WOW_PROJECT_CLASSIC
Grichelde.debug = false Grichelde.logLevel = 0 -- cannot reference Grichelde.LOG_LEVELs here as they are loaded afterwards
-- publish to global env -- publish to global env
_G.Grichelde = Grichelde _G.Grichelde = Grichelde
@@ -35,25 +32,20 @@ _G.Grichelde = Grichelde
function Grichelde:OnInitialize() function Grichelde:OnInitialize()
-- Build Interface Options window -- Build Interface Options window
self.db = self:LoadDatabase() self.db = self:LoadDatabase()
self.options, self.dialog = self:SetupOptions() self:UpgradeDatabase()
self.options, self.dialog = self:SetupOptions()
self:RefreshOptions("OnProfileChanged") self:RefreshOptions("OnProfileChanged")
self:DebugPrint(self.db.profile)
self.ldb, self.icon = self:MinimapButton()
self:SetupSlashCommands() self:SetupSlashCommands()
-- Watch for WIM and Prat to Load, then integrate
self:RegisterEvent("ADDON_LOADED", "HookIntoForOtherChatAddons")
end end
function Grichelde:OnEnable() function Grichelde:OnEnable()
-- Hook in before message is sent to replace all character occurrences where replacements have been defined in the options -- Hook in before message is sent to replace all character occurrences where replacements have been defined in the options
self:RawHook("SendChatMessage", true) self:RawHook("SendChatMessage", true)
if (_G.Misspelled) then
self:PrefixedPrint(self.L.Addon_Detected_Misspelled)
end
-- tell the world we are listening -- tell the world we are listening
self:Print(self.L.AddonLoaded, self.COLOR_CODES.PREFIX .. self.L.AddonName .. " " .. self.L.VersionAbbr .. self.version .. self.COLOR_CODES.CLOSE) self:Print(self.L.AddonLoaded, self.COLOR_CODES.PREFIX .. self.L.AddonName .. " " .. self.L.VersionAbbr .. self.version .. self.COLOR_CODES.CLOSE)
end end
@@ -64,52 +56,23 @@ end
--- Register slash commands 'gri' and 'grichelde' --- Register slash commands 'gri' and 'grichelde'
function Grichelde:SetupSlashCommands() function Grichelde:SetupSlashCommands()
local function HandleSlashCommand(input) self:RegisterChatCommand("grichelde", "HandleSlashCommand")
self:RegisterChatCommand("gri", "HandleSlashCommand")
end
function Grichelde:HandleSlashCommand(input)
-- Show the GUI if no input is supplied, otherwise handle the chat input. -- Show the GUI if no input is supplied, otherwise handle the chat input.
if self.functions.nilOrEmpty(input) then if self.functions.nilOrEmpty(input) then
LibStub("AceConfigDialog-3.0"):Open(self.name) self:OpenOptions()
else else
-- handle slash ourselves -- handle slash ourselves
self:Print("Handle slash command: " .. input) self:DebugPrint("Handle slash command: " .. input)
end if input == "mappings" then
end self:ToogleMappings()
elseif input == "options" then
self:RegisterChatCommand("grichelde", HandleSlashCommand) self:PrintOptions()
self:RegisterChatCommand("gri", HandleSlashCommand) elseif input == "profile" then
end self:PrintProfile()
--- Hook into WIM to catch whisper sending event.
-- @param event string
-- @param addonName string
function Grichelde:HookIntoForOtherChatAddons(event, addonName)
if event == "ADDON_LOADED" then
if addonName == "WIM" then
_G.WIM.RegisterWidgetTrigger("msg_box", "whisper,chat,w2w", "OnEnterPressed", Grichelde.EditBox_OnEnterPressed)
-- If available use the WIM API
if (_G.WIM.RegisterPreSendFilterText) then -- avoid error if WIM not up to date.
_G.WIM.RegisterPreSendFilterText(function(text)
return self:CheckAndReplace(text)
end)
else
-- WIM sends its chat messages via the API ChatThrottleLib, which itself hooks the default SendChatMessage api
-- many times before Grichelde will. ChatThrottleLib might potentially load before Grichelde, so we just hook
-- into ChatThrottleLib to be on the safe side.
if (_G.ChatThrottleLib) then
self.hooks["ChatThrottleLib"] = _G.ChatThrottleLib.SendChatMessage
function _G.ChatThrottleLib:SendChatMessage(prio, prefix, text, ...)
Grichelde:DebugPrint("ChatThrottleLib:SendChatMessage : Hook called")
local replacedText = Grichelde:CheckAndReplace(text)
return Grichelde.hooks["ChatThrottleLib"](_G.ChatThrottleLib, prio, prefix, replacedText, ...)
end
end
end
if (_G.WIM) then
self:PrefixedPrint(self.L.Addon_Detected_WIM)
end
end end
end end
end end

View File

@@ -1,19 +1,20 @@
## Interface: 11304 ## Interface: 11304
## Title: Grichelde ## Title: Grichelde
## Notes: Replaces characters you type in the chat box ## Notes: Replaces characters from the chat box
## Notes-de: Ersetzt eingegebene Zeichen in der Chat-Zeile ## Notes-de: Ersetzt eingegebene Zeichen in der Chat-Zeile
## Version: 0.5.0 ## Version: 0.7.2-beta
## Author: Teilzeit-Jedi ## Author: Teilzeit-Jedi
## eMail: tj@teilzeit-jedi.de ## eMail: tj@teilzeit-jedi.de
## X-Build: Classic ## X-Build: Classic
## X-Curse-Project-ID: 385480 ## X-Curse-Project-ID: 385480
## X-License: GPLv3
## X-Category: Chat/Communication ## X-Category: Chat/Communication
## X-Credits: Teilzeit-Jedi, Nathan Pieper ## X-Credits: Teilzeit-Jedi
## X-Embeds: Ace3 ## X-Embeds: Ace3
## OptionalDeps: Ace3 ## OptionalDeps: Ace3, LibDataBroker, LibIcon
## SavedVariables: GricheldeDB ## SavedVariables: GricheldeDB
libs.xml libs.xml
@@ -23,5 +24,6 @@ Grichelde.lua
GricheldeConstants.lua GricheldeConstants.lua
GricheldeUtils.lua GricheldeUtils.lua
GricheldeDatabase.lua GricheldeDatabase.lua
GricheldeUpgrade.lua
GricheldeOptions.lua GricheldeOptions.lua
GricheldeChat.lua GricheldeChat.lua

View File

@@ -2,8 +2,9 @@
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde
local nilOrEmpty, ipairs, tContains, tFilter, tInsert, tConcat, find, sub, gsub, toLower, trim, length local IsAddOnLoaded, nilOrEmpty, ipairs, spairs, tContains, tFilter, tInsert, tConcat, find, sub, isUpper, isLower, toUpper, toLower, trim, length
= Grichelde.functions.nilOrEmpty, Grichelde.functions.ipairs, Grichelde.functions.tContains, Grichelde.functions.tFilter, Grichelde.functions.tInsert, Grichelde.functions.tConcat, Grichelde.functions.find, Grichelde.functions.sub, Grichelde.functions.gsub, Grichelde.functions.toLower, Grichelde.functions.trim, Grichelde.functions.length = Grichelde.functions.IsAddOnLoaded, Grichelde.functions.nilOrEmpty, Grichelde.functions.ipairs, Grichelde.functions.spairs, Grichelde.functions.tContains, Grichelde.functions.tFilter, Grichelde.functions.tInsert, Grichelde.functions.tConcat,
Grichelde.functions.find, Grichelde.functions.sub, Grichelde.functions.isUpper, Grichelde.functions.isLower, Grichelde.functions.toUpper, Grichelde.functions.toLower, Grichelde.functions.trim, Grichelde.functions.length
--- Before a chat message is sent, check if replacement is required and replace the text accordingly. --- Before a chat message is sent, check if replacement is required and replace the text accordingly.
-- @param message string -- @param message string
@@ -13,7 +14,6 @@ local nilOrEmpty, ipairs, tContains, tFilter, tInsert, tConcat, find, sub, gsub,
function Grichelde:SendChatMessage(message, type, language, channel, ...) function Grichelde:SendChatMessage(message, type, language, channel, ...)
local replacedText = self:CheckAndReplace(message, type) local replacedText = self:CheckAndReplace(message, type)
self:DebugPrint("SendChatMessage : replacedText:", replacedText)
-- Send text in chunks if length exceeds 255 bytes after replacement -- Send text in chunks if length exceeds 255 bytes after replacement
local chunks = self:SplitText(replacedText) local chunks = self:SplitText(replacedText)
@@ -27,7 +27,7 @@ end
function Grichelde:CheckAndReplace(message, type) function Grichelde:CheckAndReplace(message, type)
local text = message local text = message
if (self:CheckReplacement(text, type)) then if (self:CheckReplacement(text, type)) then
if (_G.Misspelled) then if (IsAddOnLoaded("Misspelled")) then
self:DebugPrint("Misspelled detected: cleansing message") self:DebugPrint("Misspelled detected: cleansing message")
text = _G.Misspelled:RemoveHighlighting(text) text = _G.Misspelled:RemoveHighlighting(text)
end end
@@ -55,7 +55,7 @@ function Grichelde:CheckReplacement(text, channel)
) )
self:DebugPrint("CheckReplacement : allowed channels:", tConcat(allowedChannels, ", ")) self:DebugPrint("CheckReplacement : allowed channels:", tConcat(allowedChannels, ", "))
local type = self:ConvertBlizChannelToType(channel) local type = self:ConvertBlizChannelToType(channel)
if (type ~= nil and not tContains(allowedChannels, type)) then if (type == nil or not tContains(allowedChannels, type)) then
self:DebugPrint("CheckReplacement : skip channel type:", type) self:DebugPrint("CheckReplacement : skip channel type:", type)
return false return false
end end
@@ -103,7 +103,7 @@ function Grichelde:ReplaceText(text)
while current <= length(newText) do while current <= length(newText) do
local currentChar = sub(newText, current, current) local currentChar = sub(newText, current, current)
self:DebugPrint("current/char : %s,%s", current, currentChar) self:TracePrint("current/char : %s,%s", current, currentChar)
if ( not tContains(lookAheads, currentChar)) then if ( not tContains(lookAheads, currentChar)) then
current = current + 1 current = current + 1
@@ -153,6 +153,7 @@ function Grichelde:CheckForPreversableText(text)
"|[Cc]%x%x%x%x%x%x%x%x.-|r", -- colored items (or links) "|[Cc]%x%x%x%x%x%x%x%x.-|r", -- colored items (or links)
"|H.-|h", -- item links (http://www.wowwiki.com/ItemLink) "|H.-|h", -- item links (http://www.wowwiki.com/ItemLink)
"|T.-|t", -- textures "|T.-|t", -- textures
"|K.-|k", -- Battle.net
"|n", -- newline "|n", -- newline
"{rt[1-8]}", -- rumbered raid target icons "{rt[1-8]}", -- rumbered raid target icons
@@ -194,17 +195,188 @@ end
-- @param text string -- @param text string
-- @return string -- @return string
function Grichelde:ReplaceCharacters(text) function Grichelde:ReplaceCharacters(text)
-- todo: read from options local replacements = self.db.profile.replacements or {}
-- todo: case (in)sensitivity self:DebugPrint("ReplaceCharacters : replacements")
-- todo: consolidate consecutive self:DebugPrint(replacements)
-- todo: prevent infinite loops - is that even possible?
local replacement = text local result = text
replacement = gsub(replacement, "s", "ch") local consolidate = {}
replacement = gsub(replacement, "S", "Ch")
replacement = gsub(replacement, "t", "k") -- replacements are done first
replacement = gsub(replacement, "T", "K") for replName, replTable in spairs(replacements) do
self:DebugPrint("ReplaceCharacters : replaced \"%s\" with \"%s\"", text, replacement) local before = result
return replacement local search = replTable.searchText
if not nilOrEmpty(search) then
local replace = replTable.replaceText
consolidate[replName] = {}
if replTable.exactCase then
-- exact case
self:DebugPrint("ReplaceCharacters : \"%s => %s\" (exact case)", search, replace)
local pos, offset = 1, 0
local oldResult = result
local pos1, pos2 = find(oldResult, search, pos)
while (pos1 and pos2 and pos1 <= pos2) do
self:TracePrint("pos1: %d, pos2: %d", pos1, pos2)
local pre = sub(result, 1, pos1 - 1 + offset)
local post = sub(result, pos2 + 1 + offset)
self:TracePrint("pre: %s, post: %s", pre, post)
-- actual replacement
result = pre .. replace .. post
self:DebugPrint("result: %s", result)
-- remember positions for consolidate
if replTable.consolidate then
tInsert(consolidate[replName], pos1 + offset)
end
-- replacement text can lengthen or shorten the resulting text
-- after replacement result and lowerResult can have different sizes
offset = offset + length(replace) - length(search)
-- update values for next iteration
pos = pos2 + 1
pos1, pos2 = find(oldResult, search, pos)
end
else
self:DebugPrint("ReplaceCharacters : \"%s => %s\" (ignoreCase)", search, replace)
local pos, offset = 1, 0
local lowerResult = toLower(result)
local lowerSearch = toLower(search)
local pos1, pos2 = find(lowerResult, lowerSearch, pos)
while (pos1 and pos2 and pos1 <= pos2) do
self:TracePrint("pos1: %d, pos2: %d", pos1, pos2)
local pre = sub(result, 1, pos1 - 1 + offset)
local match = sub(result, pos1 + offset, pos2 + offset)
local post = sub(result, pos2 + 1 + offset)
self:TracePrint("pre: %s, match: %s, post: %s", pre, match, post)
-- keep cases
local repl = ""
local lastCase = nil
for p = pos1, pos2 do
self:TracePrint("p: %d", p)
local c = sub(match, p - pos1 + 1, p - pos1 + 1)
local r = sub(replace, p - pos1 + 1, p - pos1 + 1) or ""
if (isUpper(c)) then -- all UPPER-CASE letter
lastCase = true
repl = repl .. toUpper(r)
elseif (isLower(match)) then -- all lower_case letter
lastCase = false
repl = repl .. toLower(r)
else -- no letter
lastCase = nil
repl = repl .. r
end
self:TracePrint("character: %s, %s", c, r)
end
self:TracePrint("length %d > %d", length(replace), pos2 - pos1 + 1)
if (length(replace) > pos2 - pos1 + 1) then
local remainingReplace = sub(replace, pos2 - pos1 + 2)
local nextLetter = sub(post, 1, 1)
self:TracePrint("rest: %s, n: %s, lastCase: %s", remainingReplace, nextLetter, lastCase)
if (isUpper(nextLetter)) then
if lastCase == nil or lastCase == false then
repl = repl .. remainingReplace
else
repl = repl .. toUpper(remainingReplace)
end
elseif (isLower(nextLetter)) then
if lastCase == nil or lastCase == true then
repl = repl .. remainingReplace
else
repl = repl .. toLower(remainingReplace)
end
else
-- no letter
if lastCase == nil then
repl = repl .. remainingReplace
elseif lastCase == false then
repl = repl .. toLower(remainingReplace)
else
repl = repl .. toUpper(remainingReplace)
end
end
end
-- actual replacement
result = pre .. repl .. post
self:DebugPrint("result: %s", result)
-- remember positions for consolidate
if replTable.consolidate then
tInsert(consolidate[replName], pos1 + offset)
end
-- replacement text can be longer or shorter the resulting text
-- after replacement result and lowerResult can have different sizes
offset = offset + length(repl) - length(lowerSearch)
-- update values for next iteration
pos = pos2 + 1
pos1, pos2 = find(lowerResult, lowerSearch, pos)
end
end
if before ~= result then
self:DebugPrint("ReplaceCharacters : replaced \"%s\" with \"%s\"", before, result)
end
else
self:DebugPrint("ReplaceCharacters : Skip replacement for empty mapping")
end
end
-- consolidation is done last
for replName, replTable in spairs(replacements) do
local before = result
--local search = replTable.searchText
local search = replTable.searchText
if not nilOrEmpty(search) then
local replace = replTable.replaceText
local lowerResult = toLower(result)
local offset = 0
if replTable.consolidate then
self:DebugPrint("ReplaceCharacters : consolidating \"%s => %s\"", search, replace)
self:DebugPrint("consolidate[" .. replName .. "] is:")
self:DebugPrint(consolidate[replName])
for _, pos1 in spairs(consolidate[replName]) do
local pos2 = pos1 + length(replace) - 1
self:TracePrint("pos1: %d, pos2: %d", pos1, pos2)
local match = sub(lowerResult, pos1, pos2)
local next = sub(lowerResult, pos2 + 1, pos2 + 1 + pos2 - pos1)
self:TracePrint("match: %s, next: %s", match, next)
local _, p2 = find(next, "^" .. match)
self:TracePrint("p2: %d", p2)
if (p2) then
result = sub(result, 1, pos2 + offset) .. sub(result, pos2 + 1 + p2 + offset)
-- consolidation will shorten the resulting text
offset = offset + length(result) - length(lowerResult)
end
self:DebugPrint("result: %s", result)
end
end
if before ~= result then
self:DebugPrint("ReplaceCharacters : consolidate \"%s\" with \"%s\"", before, result)
end
else
self:DebugPrint("ReplaceCharacters : Skip consolidation for empty mapping")
end
end
self:DebugPrint("ReplaceCharacters : final text:", result)
return result
end end
--- Splits a long text in longest possible chunks of <= 255 length, split at last available space --- Splits a long text in longest possible chunks of <= 255 length, split at last available space

View File

@@ -2,21 +2,38 @@
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde
-- upvalues and constants -- constants and upvalues
Grichelde.LOG_LEVEL = {}
Grichelde.LOG_LEVEL.DEBUG = 1
Grichelde.LOG_LEVEL.TRACE = 2
Grichelde.MAPPING_OFFSET = 10
Grichelde.ICONS = {}
Grichelde.ICONS.MOVE_UP = "Interface\\MainMenuBar\\UI-MainMenu-ScrollUpButton-Up"
Grichelde.ICONS.MOVE_DOWN = "Interface\\MainMenuBar\\UI-MainMenu-ScrollDownButton-Up"
-- colors: -- colors:
Grichelde.COLORS = {}
Grichelde.COLORS.NORMAL = _G.NORMAL_FONT_COLOR
Grichelde.COLORS.HIGHLIGHT = _G.HIGHLIGHT_FONT_COLOR
Grichelde.COLORS.RED = _G.RED_FONT_COLOR
Grichelde.COLORS.GREEN = _G.GREEN_FONT_COLOR
Grichelde.COLOR_CODES = {} Grichelde.COLOR_CODES = {}
Grichelde.COLOR_CODES.PREFIX = "|c00FFAA00" Grichelde.COLOR_CODES.PREFIX = "|c00FFAA00"
-- https://github.com/stoneharry/Misc-WoW-Stuff/blob/master/EoC%20Interface/FrameXML/Constants.lua -- https://github.com/stoneharry/Misc-WoW-Stuff/blob/master/EoC%20Interface/FrameXML/Constants.lua
Grichelde.COLOR_CODES.NORMAL = _G.NORMAL_FONT_COLOR_CODE or "|cffffd200"; Grichelde.COLOR_CODES.NORMAL = _G.NORMAL_FONT_COLOR_CODE or "|cffffd200"
Grichelde.COLOR_CODES.HIGHLIGHT = _G.HIGHLIGHT_FONT_COLOR_CODE or "|cffffffff"; Grichelde.COLOR_CODES.HIGHLIGHT = _G.HIGHLIGHT_FONT_COLOR_CODE or "|cffffffff"
Grichelde.COLOR_CODES.RED = _G.RED_FONT_COLOR_CODE or "|cffff2020"; Grichelde.COLOR_CODES.RED = _G.RED_FONT_COLOR_CODE or "|cffff2020"
Grichelde.COLOR_CODES.GREEN = _G.GREEN_FONT_COLOR_CODE or "|cff20ff20"; Grichelde.COLOR_CODES.GREEN = _G.GREEN_FONT_COLOR_CODE or "|cff20ff20"
Grichelde.COLOR_CODES.GRAY = _G.GRAY_FONT_COLOR_CODE or "|cff808080"; Grichelde.COLOR_CODES.LIGHTGRAY = "|cffC0C0C0"
Grichelde.COLOR_CODES.YELLOW = _G.YELLOW_FONT_COLOR_CODE or "|cffffff00"; Grichelde.COLOR_CODES.GRAY = _G.GRAY_FONT_COLOR_CODE or "|cff808080"
Grichelde.COLOR_CODES.LIGHTYELLOW = _G.LIGHTYELLOW_FONT_COLOR_CODE or "|cffffff9a"; Grichelde.COLOR_CODES.DARKGRAY = "|cff404040"
Grichelde.COLOR_CODES.ORANGE = _G.ORANGE_FONT_COLOR_CODE or "|cffff7f3f"; Grichelde.COLOR_CODES.YELLOW = _G.YELLOW_FONT_COLOR_CODE or "|cffffff00"
Grichelde.COLOR_CODES.CLOSE = _G.FONT_COLOR_CODE_CLOSE or "|r"; Grichelde.COLOR_CODES.LIGHTYELLOW = _G.LIGHTYELLOW_FONT_COLOR_CODE or "|cffffff9a"
Grichelde.COLOR_CODES.ORANGE = _G.ORANGE_FONT_COLOR_CODE or "|cffff7f3f"
Grichelde.COLOR_CODES.CLOSE = _G.FONT_COLOR_CODE_CLOSE or "|r"
Grichelde.slashCommands = { "/s", "/say", "/e", "/em", "/me", "/emote", "/y", "/yell", "/sh", "/shout", "/p", "/party", "/pl", "/partyleader", "/g", "/gc", "/guild", "/o", "/osay", "/officer", "/raid", "/rsay", "/rl", "/raidleader", "/rw", "/raidwarning", "/i", "/instance", "/bg", "/battleground", "/w", "/whisper", "/t", "/tell", "/send", "/r", "/reply" } Grichelde.slashCommands = { "/s", "/say", "/e", "/em", "/me", "/emote", "/y", "/yell", "/sh", "/shout", "/p", "/party", "/pl", "/partyleader", "/g", "/gc", "/guild", "/o", "/osay", "/officer", "/raid", "/rsay", "/rl", "/raidleader", "/rw", "/raidwarning", "/i", "/instance", "/bg", "/battleground", "/w", "/whisper", "/t", "/tell", "/send", "/r", "/reply" }
@@ -35,6 +52,7 @@ local function spairs(t , orderFunc)
if orderFunc then if orderFunc then
Grichelde.functions.tSort(sortedKeys, function(a, b) return orderFunc(sortedKeys, a, b) end) Grichelde.functions.tSort(sortedKeys, function(a, b) return orderFunc(sortedKeys, a, b) end)
else else
-- lexicographical order
Grichelde.functions.tSort(sortedKeys) Grichelde.functions.tSort(sortedKeys)
end end
@@ -44,6 +62,8 @@ local function spairs(t , orderFunc)
it = it + 1 it = it + 1
if sortedKeys[it] then if sortedKeys[it] then
return sortedKeys[it], t[sortedKeys[it]] return sortedKeys[it], t[sortedKeys[it]]
else
return nil
end end
end end
end end
@@ -86,8 +106,31 @@ local function tClone(orig)
return copy return copy
end end
local function isChar(word)
return Grichelde.functions.find(word, "%a+")
end
local function isNumber(digit)
return Grichelde.functions.find(digit, "%d+")
end
local function isUpper(word)
return Grichelde.functions.isChar(word) and word == Grichelde.functions.toUpper(word)
end
local function isLower(word)
return Grichelde.functions.isChar(word) and word == Grichelde.functions.toLower(word)
end
local function isCapital(word)
return Grichelde.functions.legnth(word) > 1
and Grichelde.functions.isUpper(Grichelde.functions.sub(word,1,1))
and Grichelde.functions.isLower(Grichelde.functions.sub(word,2))
end
-- faster function lookups by mapping to local refs -- faster function lookups by mapping to local refs
Grichelde.functions = {} Grichelde.functions = {}
Grichelde.functions.IsAddOnLoaded = _G.IsAddOnLoaded
Grichelde.functions.type = _G.type Grichelde.functions.type = _G.type
Grichelde.functions.print = _G.print Grichelde.functions.print = _G.print
Grichelde.functions.nilOrEmpty = nilOrEmpty Grichelde.functions.nilOrEmpty = nilOrEmpty
@@ -112,8 +155,13 @@ Grichelde.functions.gsub = _G.string.gsub
Grichelde.functions.match = _G.strmatch Grichelde.functions.match = _G.strmatch
Grichelde.functions.join = _G.strjoin Grichelde.functions.join = _G.strjoin
Grichelde.functions.split = _G.strsplit Grichelde.functions.split = _G.strsplit
Grichelde.functions.toLower = _G.strlower
Grichelde.functions.toUpper = _G.strupper Grichelde.functions.toUpper = _G.strupper
Grichelde.functions.toLower = _G.strlower
Grichelde.functions.isChar = isChar
Grichelde.functions.isNumber = isNumber
Grichelde.functions.isUpper = isUpper
Grichelde.functions.isLower = isLower
Grichelde.functions.isCapital = isCapital
Grichelde.functions.format = _G.string.format Grichelde.functions.format = _G.string.format
Grichelde.functions.rep = _G.string.rep Grichelde.functions.rep = _G.string.rep
Grichelde.functions.trim = _G.strtrim Grichelde.functions.trim = _G.strtrim

View File

@@ -2,13 +2,17 @@
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde
local pairs, ipairs, tInsert, tSort, unpack, join, toString local pairs, tInsert, tClone, unpack, join, toString
= Grichelde.functions.pairs, Grichelde.functions.ipairs, Grichelde.functions.tInsert, Grichelde.functions.tSort, Grichelde.functions.unpack, Grichelde.functions.join, Grichelde.functions.toString = Grichelde.functions.pairs, Grichelde.functions.tInsert, Grichelde.functions.tClone, Grichelde.functions.unpack, Grichelde.functions.join, Grichelde.functions.toString
local defaultConfig = { function Grichelde:GetDefaultConfig()
return {
global = {}, global = {},
profile = { profile = {
enabled = true, enabled = true,
minimapButton = {
hide = false
},
channels = { channels = {
["*"] = false, ["*"] = false,
say = true, say = true,
@@ -23,29 +27,37 @@ local defaultConfig = {
order = 9999, order = 9999,
searchText = "", searchText = "",
replaceText = "", replaceText = "",
caseSensitive = false, exactCase = false,
consolidate = true, consolidate = true,
}, },
replacement_0 = { replacement_10 = {
order = 5, order = 10,
searchText = "s", searchText = "s",
replaceText = "ch", replaceText = "ch",
caseSensitive = false, exactCase = false,
consolidate = true, consolidate = true,
}, },
replacement_1 = { replacement_11 = {
order = 9, order = 11,
searchText = "t", searchText = "t",
replaceText = "ck", replaceText = "ck",
caseSensitive = false, exactCase = false,
consolidate = true,
},
replacement_12 = {
order = 12,
searchText = "Zark",
replaceText = "toter Schamane",
exactCase = false,
consolidate = true, consolidate = true,
} }
} }
} }
} }
end
function Grichelde:LoadDatabase() function Grichelde:LoadDatabase()
local db = LibStub("AceDB-3.0"):New(self.name .."DB", defaultConfig, true) local db = LibStub("AceDB-3.0"):New(self.name .."DB", self:GetDefaultConfig(), true)
db.RegisterCallback(self, "OnNewProfile", "RefreshOptions") db.RegisterCallback(self, "OnNewProfile", "RefreshOptions")
db.RegisterCallback(self, "OnProfileChanged", "RefreshOptions") db.RegisterCallback(self, "OnProfileChanged", "RefreshOptions")
@@ -57,10 +69,17 @@ function Grichelde:LoadDatabase()
end end
function Grichelde:SyncToDatabase(info, val) function Grichelde:SyncToDatabase(info, val)
self:TracePrint("SyncToDatabase : info")
for i = 0, #info do
self:TracePrint("%d = %s", i, info[i])
end
local option = self.db.profile local option = self.db.profile
local path = 1 local path = 1
while (path < #info) do while (path < #info) do
if info[path] ~= "mappings" then
option = option[info[path]] -- or nil option = option[info[path]] -- or nil
end
path = path + 1 path = path + 1
end end
local optionPath = join(".", unpack(info, 1, #info)) local optionPath = join(".", unpack(info, 1, #info))
@@ -69,10 +88,17 @@ function Grichelde:SyncToDatabase(info, val)
end end
function Grichelde:ReadFromDatabase(info) function Grichelde:ReadFromDatabase(info)
self:TracePrint("ReadFromDatabase : info")
for i = 0, #info do
self:TracePrint("%d = %s", i, info[i])
end
local option = self.db.profile local option = self.db.profile
local path = 1 local path = 1
while (path <= #info) do while (path <= #info) do
if info[path] ~= "mappings" then
option = option[info[path]] -- or nil option = option[info[path]] -- or nil
end
path = path + 1 path = path + 1
end end
local optionPath = join(".", unpack(info, 1, #info)) local optionPath = join(".", unpack(info, 1, #info))
@@ -80,43 +106,54 @@ function Grichelde:ReadFromDatabase(info)
return option return option
end end
--- Sorts a replacements table by order sub-field. --- Sorts a replacements table by order sub-field and rename.
--- Usually called with with self.db.profile.replacements --- Do NOT reassign self.db.profile.replacements here or with its output as it will break defaults
-- @param replacementsTable table function Grichelde:ReorderReplacements()
-- @return table local replacements = self.db.profile.replacements or {}
function Grichelde:ReorderReplacements(replacementsTable)
local replacements = replacementsTable or {} self:TracePrint("ReorderReplacements : unsorted table")
local sortedByOrder = {} self:TracePrint(replacements)
for replName, _ in pairs(replacements) do
tInsert(sortedByOrder, replName) local orderToName = {}
local size = 0
for replName, replTable in pairs(replacements) do
size = size + 1
tInsert(orderToName, replTable.order, replName)
end end
tSort(sortedByOrder) -- lexicographical order will do for non-nil values self:TracePrint("ReorderReplacements : size: %d, orderToName", size)
--[[tSort(sortedByOrder, function(a, b) self:TracePrint(orderToName)
self:DebugPrint("ReorderReplacements : sort ", a, b)
if a then
if b then
return a < b
else
return a
end
else
return b
end
end)]]
self:DebugPrint("ReorderReplacements : sortedByOrder") local sorted = {}
self:DebugPrint(sortedByOrder) local index, count = 0, 0
local sortedReplacements = {} while count < size do
local index = 0 local replName = orderToName[index]
for _, replName in ipairs(sortedByOrder) do if replName and replacements[replName] then
sortedReplacements["replacement_"..index] = replacements[replName] self:TracePrint("ReorderReplacements : replName: %s, replTable", replName)
sortedReplacements["replacement_"..index].order = index self:TracePrint(replacements[replName])
local order = Grichelde.MAPPING_OFFSET + count
sorted["replacement_" .. order] = tClone(replacements[replName])
sorted["replacement_" .. order].order = order
count = count + 1
end
index = index + 1 index = index + 1
if ( index > 10000) then break end
end end
--self:DebugPrint("ReorderReplacements : sorted table") -- self:TracePrint("ReorderReplacements : sorted")
--self:DebugPrint(sortedReplacements) -- self:TracePrint(sorted)
return sortedReplacements
-- do NOT set self.db.profile.replacements = {} it will break defaults
for replName, _ in pairs(replacements) do
replacements[replName] = nil
end
-- copy over sorted replacements
for replName, replTable in pairs(sorted) do
replacements[replName] = replTable
end
self:DebugPrint("ReorderReplacements : sorted table")
self:DebugPrint(self.db.profile.replacements)
end end

View File

@@ -2,8 +2,8 @@
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde
local nilOrEmpty, pairs, tSize, unpack, find, join, toString, toNumber local nilOrEmpty, pairs, find, match, toString, toNumber
= Grichelde.functions.nilOrEmpty, Grichelde.functions.pairs, Grichelde.functions.tSize, Grichelde.functions.unpack, Grichelde.functions.find, Grichelde.functions.join, Grichelde.functions.toString, Grichelde.functions.toNumber = Grichelde.functions.nilOrEmpty, Grichelde.functions.pairs, Grichelde.functions.find, Grichelde.functions.match, Grichelde.functions.toString, Grichelde.functions.toNumber
function Grichelde:CreateOptionsUI() function Grichelde:CreateOptionsUI()
return { return {
@@ -22,78 +22,96 @@ function Grichelde:CreateOptionsUI()
desc = self:Format(self.L.Options_Enabled_Desc, self.L.AddonName), desc = self:Format(self.L.Options_Enabled_Desc, self.L.AddonName),
disabled = false, disabled = false,
}, },
minimapButton = {
order = 1,
type = "toggle",
name = self.L.Options_Minimap_Button_Name,
desc = self.L.Options_Minimap_Button_Desc,
set = function(info, val) self:ToogleMinimapButton(info, val) end,
get = function(info) return not self.db.profile.minimapButton.hide end,
disabled = false,
},
channels = { channels = {
order = 2, order = 2,
type = "group", type = "group",
name = self.L.Options_Channels_Group_Name, name = self.L.Options_Channels_Group_Name,
desc = self:Format(self.L.Options_Channels_Group_Desc, self.L.AddonName), desc = self:Format(self.L.Options_Channels_Group_Desc, self.L.AddonName),
args = { args = {
say = { header = {
order = 0,
type = "toggle",
name = self.L.Options_Channels_ChannelSay_Name,
desc = self:Format(self.L.Options_Channels_ChannelSay_Desc, self.L.AddonName),
},
emote = {
order = 1, order = 1,
type = "toggle", type = "description",
name = self.L.Options_Channels_ChannelEmote_Name, name = self.L.Options_Channels_Header
desc = self:Format(self.L.Options_Channels_ChannelEmote_Desc, self.L.AddonName),
}, },
yell = { spacer = {
order = 2, order = 2,
type = "toggle", type = "header",
name = self.L.Options_Channels_ChannelYell_Name, name = ""
desc = self:Format(self.L.Options_Channels_ChannelYell_Desc, self.L.AddonName),
}, },
party = { say = {
order = 3,
type = "toggle",
name = self.L.Options_Channels_ChannelParty_Name,
desc = self:Format(self.L.Options_Channels_ChannelParty_Desc, self.L.AddonName),
},
guild = {
order = 4,
type = "toggle",
name = self.L.Options_Channels_ChannelGuild_Name,
desc = self:Format(self.L.Options_Channels_ChannelGuild_Desc, self.L.AddonName),
},
officer = {
order = 5,
type = "toggle",
name = self.L.Options_Channels_ChannelOfficer_Name,
desc = self:Format(self.L.Options_Channels_ChannelOfficer_Desc, self.L.AddonName),
},
raid = {
order = 6,
type = "toggle",
name = self.L.Options_Channels_ChannelRaid_Name,
desc = self:Format(self.L.Options_Channels_ChannelRaid_Desc, self.L.AddonName),
},
raidWarning = {
order = 7,
type = "toggle",
name = self.L.Options_Channels_ChannelRaidWarning_Name,
desc = self:Format(self.L.Options_Channels_ChannelRaidWarning_Desc, self.L.AddonName),
},
instance = {
order = 8,
type = "toggle",
name = self.L.Options_Channels_ChannelInstance_Name,
desc = self:Format(self.L.Options_Channels_ChannelInstance_Desc, self.L.AddonName),
},
battleground = {
order = 9,
type = "toggle",
name = self.L.Options_Channels_ChannelBattleground_Name,
desc = self:Format(self.L.Options_Channels_ChannelBattleground_Desc, self.L.AddonName),
},
whisper = {
order = 10, order = 10,
type = "toggle", type = "toggle",
name = self.L.Options_Channels_ChannelWhisper_Name, name = self.L.Options_Channel_Say_Name,
desc = self:Format(self.L.Options_Channels_ChannelWhisper_Desc, self.L.AddonName), desc = self:Format(self.L.Options_Channel_Say_Desc, self.L.AddonName),
},
emote = {
order = 11,
type = "toggle",
name = self.L.Options_Channel_Emote_Name,
desc = self:Format(self.L.Options_Channel_Emote_Desc, self.L.AddonName),
},
yell = {
order = 12,
type = "toggle",
name = self.L.Options_Channel_Yell_Name,
desc = self:Format(self.L.Options_Channel_Yell_Desc, self.L.AddonName),
},
party = {
order = 13,
type = "toggle",
name = self.L.Options_Channel_Party_Name,
desc = self:Format(self.L.Options_Channel_Party_Desc, self.L.AddonName),
},
guild = {
order = 14,
type = "toggle",
name = self.L.Options_Channel_Guild_Name,
desc = self:Format(self.L.Options_Channel_Guild_Desc, self.L.AddonName),
},
officer = {
order = 15,
type = "toggle",
name = self.L.Options_Channel_Officer_Name,
desc = self:Format(self.L.Options_Channel_Officer_Desc, self.L.AddonName),
},
raid = {
order = 16,
type = "toggle",
name = self.L.Options_Channel_Raid_Name,
desc = self:Format(self.L.Options_Channel_Raid_Desc, self.L.AddonName),
},
raidWarning = {
order = 17,
type = "toggle",
name = self.L.Options_Channel_RaidWarning_Name,
desc = self:Format(self.L.Options_Channel_RaidWarning_Desc, self.L.AddonName),
},
instance = {
order = 18,
type = "toggle",
name = self.L.Options_Channel_Instance_Name,
desc = self:Format(self.L.Options_Channel_Instance_Desc, self.L.AddonName),
},
battleground = {
order = 19,
type = "toggle",
name = self.L.Options_Channel_Battleground_Name,
desc = self:Format(self.L.Options_Channel_Battleground_Desc, self.L.AddonName),
},
whisper = {
order = 20,
type = "toggle",
name = self.L.Options_Channel_Whisper_Name,
desc = self:Format(self.L.Options_Channel_Whisper_Desc, self.L.AddonName),
}, },
} }
}, },
@@ -110,7 +128,7 @@ function Grichelde:CreateOptionsUI()
confirm = false, confirm = false,
name = self.L.Options_Replacements_Add_Name, name = self.L.Options_Replacements_Add_Name,
desc = self.L.Options_Replacements_Add_Desc, desc = self.L.Options_Replacements_Add_Desc,
func = function(info) self:AddReplacement(info) end func = function(info) self:AddEmptyMapping(info) end
}, },
deleteAll = { deleteAll = {
order = 1, order = 1,
@@ -119,55 +137,93 @@ function Grichelde:CreateOptionsUI()
confirmText = self.L.Options_Replacements_DeleteAll_ConfirmText, confirmText = self.L.Options_Replacements_DeleteAll_ConfirmText,
name = self.L.Options_Replacements_DeleteAll_Name, name = self.L.Options_Replacements_DeleteAll_Name,
desc = self.L.Options_Replacements_DeleteAll_Desc, desc = self.L.Options_Replacements_DeleteAll_Desc,
func = function(info) self:DeleteAllReplacements(info) end func = function(info) self:DeleteAllMappings(info) end
} },
header = {
order = 3,
type = "description",
name = self.L.Options_Replacements_Header
},
spacer = {
order = 4,
type = "header",
name = ""
},
} }
} }
} }
} }
end end
function Grichelde:CreateReplacement(offset) function Grichelde:CreateMapping(offset)
return { return {
order = offset or 9999, order = offset or 9999,
type = "group", type = "group",
name = function(info) return self:MappingName(info) end, name = function(info) return self:MappingName(info) end,
desc = self.L.Options_Replacement_Group_Desc, desc = self.L.Options_Mapping_Group_Desc,
childGroups = "tree", childGroups = "tree",
args = { args = {
searchText = { searchText = {
order = 0, order = 0,
type = "input", type = "input",
name = self.L.Options_Replacement_SearchText_Name, name = self.L.Options_Mapping_SearchText_Name,
desc = self.L.Options_Replacement_SearchText_Desc, desc = self.L.Options_Mapping_SearchText_Desc,
}, },
replaceText = { replaceText = {
order = 1, order = 1,
type = "input", type = "input",
name = self.L.Options_Replacement_ReplaceText_Name, name = self.L.Options_Mapping_ReplaceText_Name,
desc = self.L.Options_Replacement_ReplaceText_Desc, desc = self.L.Options_Mapping_ReplaceText_Desc,
}, },
caseSensitive = { exactCase = {
order = 2, order = 2,
type = "toggle", type = "toggle",
name = self.L.Options_Replacement_CaseSensitive_Name, name = self.L.Options_Mapping_ExactCase_Name,
desc = self.L.Options_Replacement_CaseSensitive_Desc, desc = self.L.Options_Mapping_ExactCase_Desc,
width = "full",
}, },
consolidate = { consolidate = {
order = 3, order = 3,
type = "toggle", type = "toggle",
name = self.L.Options_Replacement_Consolidate_Name, name = self.L.Options_Mapping_Consolidate_Name,
desc = self.L.Options_Replacement_Consolidate_Desc, desc = self.L.Options_Mapping_Consolidate_Desc,
width = 2 width = "full"
},
moveUp = {
order = 10,
type = "execute",
-- name = self.L.Options_Mapping_MoveUp_Name,
name = "",
desc = self.L.Options_Mapping_MoveUp_Desc,
image = Grichelde.ICONS.MOVE_UP,
width = 0.15,
func = function(info) self:MoveUp(info) end
},
moveDown = {
order = 11,
type = "execute",
-- name = self.L.Options_Mapping_MoveDown_Name,
name = "",
desc = self.L.Options_Mapping_MoveDown_Desc,
image = Grichelde.ICONS.MOVE_DOWN,
width = 0.15,
func = function(info) self:MoveDown(info) end
},
spacer = {
order = 18,
type = "description",
name = "",
width = 1.2,
}, },
delete = { delete = {
order = 4, order = 19,
type = "execute", type = "execute",
confirm = true, confirm = true,
confirmText = self.L.Options_Replacements_Delete_ConfirmText, confirmText = self.L.Options_Mapping_Delete_ConfirmText,
name = self.L.Options_Replacement_Delete_Name, name = self.L.Options_Mapping_Delete_Name,
desc = self.L.Options_Replacement_Delete_Desc, desc = self.L.Options_Mapping_Delete_Desc,
func = function(info) self:DeleteReplacement(info) end width = 0.5,
func = function(info) self:DeleteMapping(info) end
}, },
} }
} }
@@ -187,28 +243,6 @@ function Grichelde:SetupOptions()
return options, dialog return options, dialog
end end
function Grichelde:IsDisabled(info)
if info.option.type == "group" then
return false
end
return not self.db.profile.enabled
end
function Grichelde:MappingName(info)
local option = self.db.profile
local path = 1
while (path <= #info) do
option = option[info[path]]
path = path + 1
end
if nilOrEmpty(option.searchText) and nilOrEmpty(option.replaceText) then
return self.L.Options_Replacement_EmptyMapping
else
return self:Format(self.L.Options_Replacement_Group_Name, option.searchText or "", option.replaceText or "")
end
end
function Grichelde:RefreshOptions(event) function Grichelde:RefreshOptions(event)
self:DebugPrint("RefreshOptions : event:", event) self:DebugPrint("RefreshOptions : event:", event)
if event == "OnNewProfile" then if event == "OnNewProfile" then
@@ -225,15 +259,97 @@ function Grichelde:RefreshOptions(event)
self:DebugPrint("Refreshing Profile %s on options change: %s", self.db:GetCurrentProfile(), event) self:DebugPrint("Refreshing Profile %s on options change: %s", self.db:GetCurrentProfile(), event)
end end
self:RefreshReplacements(self:ReorderReplacements(self.db.profile.replacements)) self:ReorderReplacements()
self:RefreshReplacements(self.db.profile.replacements)
end end
--- add Minimap button
function Grichelde:MinimapButton()
local function toggleOptionsUI(_, button)
if button == 'LeftButton' then
self:ToggleOptions()
elseif button == 'RightButton' then
self:ToogleMappings()
end
end
local function tooltip(tooltip)
if not tooltip or not tooltip.AddLine then return end
tooltip:SetText(self.L.AddonName,
Grichelde.COLORS.HIGHLIGHT.r, Grichelde.COLORS.HIGHLIGHT.g, Grichelde.COLORS.HIGHLIGHT.b, Grichelde.COLORS.HIGHLIGHT.a
)
tooltip:AddDoubleLine(self.L.Options_Minimap_Tooltip_Options_Left, self.L.Options_Minimap_Tooltip_Options_Right,
Grichelde.COLORS.GREEN.r, Grichelde.COLORS.GREEN.g, Grichelde.COLORS.GREEN.b, Grichelde.COLORS.GREEN.a,
Grichelde.COLORS.NORMAL.r, Grichelde.COLORS.NORMAL.g, Grichelde.COLORS.NORMAL.b, Grichelde.COLORS.NORMAL.a
)
tooltip:AddDoubleLine(self.L.Options_Minimap_Tooltip_Mappings_Left, self.L.Options_Minimap_Tooltip_Mappings_Right,
Grichelde.COLORS.GREEN.r, Grichelde.COLORS.GREEN.g, Grichelde.COLORS.GREEN.b, Grichelde.COLORS.GREEN.a,
Grichelde.COLORS.NORMAL.r, Grichelde.COLORS.NORMAL.g, Grichelde.COLORS.NORMAL.b, Grichelde.COLORS.NORMAL.a
)
end
local ldb = LibStub("LibDataBroker-1.1"):NewDataObject(self.name, {
type = "launcher",
text = self.AddonName,
icon = "Interface\\Icons\\Spell_Holy_Silence",
--icon = ([[Interface\Addons\%s\%s]]):format(self.name, self.name),
OnClick = toggleOptionsUI,
OnRightClick = function() self:ShowMappings() end,
OnTooltipShow = tooltip,
})
local icon = LibStub("LibDBIcon-1.0")
self:DebugPrint("MinimapButton : hidden: ", self.db.profile.minimapButton.hide)
icon:Register(self.name, ldb, self.db.profile.minimapButton)
return ldb, icon
end
function Grichelde:ToggleOptions()
self:DebugPrint("MinimapButton : options open: ", self.dialog.OpenFrames[self.name])
if self.dialog.OpenFrames[self.name] then
self:CloseOptions()
else
self:OpenOptions()
end
end
function Grichelde:OpenOptions()
self.dialog:Open(self.name)
end
function Grichelde:CloseOptions()
self.dialog:Close(self.name)
end
function Grichelde:IsDisabled(info)
if info.option.type == "group" then
return false
end
return not self.db.profile.enabled
end
function Grichelde:MappingName(info)
-- self:TracePrint("MappingName : info")
-- self:TracePrint(info)
local option = self.db.profile.replacements[info[2]]
if nilOrEmpty(option.searchText) and nilOrEmpty(option.replaceText) then
return self.L.Options_Mapping_EmptyMapping
else
return self:Format(self.L.Options_Mapping_Group_Name, option.searchText or "", option.replaceText or "")
end
end
--- Create UI options for rhe given replacement table (from DB). --- Create UI options for rhe given replacement table (from DB).
--- Usually called with with self.db.profile.replacements --- Usually called with with self.db.profile.replacements
-- @param replacementsTable -- @param replacementsTable
function Grichelde:RefreshReplacements(replacementsTable) function Grichelde:RefreshReplacements(replacementsTable)
--self:DebugPrint("RefreshReplacements : DB table:") self:TracePrint("RefreshReplacements : DB table:")
--self:DebugPrint(replacementsTable) self:TracePrint(replacementsTable)
-- remove all previous replacements from options (not DB), except header and buttons -- remove all previous replacements from options (not DB), except header and buttons
local replacements = self.options.args.replacements.args or {} local replacements = self.options.args.replacements.args or {}
@@ -245,59 +361,162 @@ function Grichelde:RefreshReplacements(replacementsTable)
for replName, _ in pairs(replacementsTable or {}) do for replName, _ in pairs(replacementsTable or {}) do
local _, replNumber = self:SplitOnFirstMatch(replName, "_") local _, replNumber = self:SplitOnFirstMatch(replName, "_")
replacements[replName] = self:CreateReplacement(toNumber(replNumber)) replacements[replName] = self:CreateMapping(toNumber(replNumber))
end end
--self:DebugPrint("RefreshReplacements : UI options:") -- self:TracePrint("RefreshReplacements : UI options:")
--self:DebugPrint(replacements) -- self:TracePrint(replacements)
self.dialog:ConfigTableChanged(nil, self.name) self.dialog:ConfigTableChanged(nil, self.name)
end end
function Grichelde:AddReplacement() function Grichelde:AddEmptyMapping()
local replacements = self.db.profile.replacements local replacements = self.db.profile.replacements or {}
local maxRepl = tSize(replacements)
local newMapping = "replacement_" .. maxRepl
self:DebugPrint("AddReplacement : new replacement key:", newMapping)
-- setting replacements[newMapping] = {} will deactivate defaults self:DebugPrint("AddEmptyMapping : old DB entries:")
replacements[newMapping].order = maxRepl self:DebugPrint(replacements)
--self:DebugPrint("AddReplacements : all DB entries:")
--self:DebugPrint(replacements)
--self.db.profile.replacements = replacements local maxRepl = Grichelde.MAPPING_OFFSET
self:RefreshOptions("AddReplacement " .. newMapping) for replName, _ in pairs(replacements) do
local num = match(replName, "^replacement_(%d+)")
if num and maxRepl < toNumber(num) then
maxRepl = toNumber(num)
end
end
local newMapping = "replacement_" .. toString(maxRepl + 1)
self:DebugPrint("AddEmptyMapping : new mapping key:", newMapping)
-- do NOT set self.db.profile.replacements = {} it will break defaults
replacements[newMapping].order = toString(maxRepl + 1) -- will be reordered anyway
self:DebugPrint("AddEmptyMapping : new DB entries:")
self:DebugPrint(replacements)
self:RefreshOptions("AddEmptyMapping " .. newMapping)
self.dialog:SelectGroup(self.name, "replacements", newMapping) self.dialog:SelectGroup(self.name, "replacements", newMapping)
end end
function Grichelde:DeleteReplacement(info) function Grichelde:MoveUp(info)
self:DebugPrint("DeleteReplacement") self:TracePrint("MoveUp : info")
local option = self.db.profile for i = 0, #info do
local path = 1 self:TracePrint("%d = %s", i, info[i])
while (path < #info - 1) do
option = option[info[path]]
--self:DebugPrint(option)
path = path + 1
end end
local optionPath = join(".", unpack(info, 1, #info))
self:DebugPrint("delete option \"%s\": %s", optionPath, toString(option[info[path]]))
option[info[path]] = nil
self:RefreshOptions("DeleteReplacement " .. info[path]) local replacements = self.db.profile.replacements or {}
local currentName = info[2]
local _, replNumber = self:SplitOnFirstMatch(info[path], "_") self:DebugPrint("MoveUp : \"%s\"", currentName)
self:DebugPrint(replacements[currentName])
local _, replNumber = self:SplitOnFirstMatch(currentName, "_")
local currentOrder = toNumber(replNumber)
-- if not on top
if currentOrder ~= Grichelde.MAPPING_OFFSET then
local swapName = "replacement_" .. toString(currentOrder - 1)
-- swap ordering
self:DebugPrint("swap with option %s", swapName)
replacements[swapName].order = currentOrder
replacements[currentName].order = currentOrder - 1
self:RefreshOptions("MoveUp " .. currentName)
self:DebugPrint("MoveUp : refresh focus on %s", swapName)
self.dialog:SelectGroup(self.name, "replacements", swapName)
else
self:DebugPrint("MoveUp : already on top")
end
end
function Grichelde:MoveDown(info)
self:TracePrint("MoveDown : info")
for i = 0, #info do
self:TracePrint("%d = %s", i, info[i])
end
local replacements = self.db.profile.replacements or {}
local currentName = info[2]
self:DebugPrint("MoveDown : \"%s\"", currentName)
self:DebugPrint(replacements[currentName])
local _, replNumber = self:SplitOnFirstMatch(currentName, "_")
local currentOrder = toNumber(replNumber)
local maxRepl = Grichelde.MAPPING_OFFSET
for replName, _ in pairs(replacements) do
local num = match(replName, "^replacement_(%d+)")
if num and maxRepl < toNumber(num) then
maxRepl = toNumber(num)
end
end
-- if not last element
self:DebugPrint("MoveDown : maxRepl: %d", maxRepl)
if currentOrder < maxRepl then
local swapName = "replacement_" .. toString(currentOrder + 1)
-- swap ordering
self:DebugPrint("swap with option %s", swapName)
replacements[swapName].order = currentOrder
replacements[currentName].order = currentOrder + 1
self:RefreshOptions("MoveDown " .. currentName)
self:DebugPrint("MoveDown : refresh focus on %s", swapName)
self.dialog:SelectGroup(self.name, "replacements", swapName)
else
self:DebugPrint("MoveDown : already at bottom")
end
end
function Grichelde:DeleteMapping(info)
self:TracePrint("DeleteMapping : info")
for i = 0, #info do
self:TracePrint("%d = %s", i, info[i])
end
local currentName = info[2]
self:DebugPrint("delete option: %s", currentName)
self.db.profile.replacements[currentName] = nil
self:RefreshOptions("DeleteMapping " .. currentName)
local _, replNumber = self:SplitOnFirstMatch(currentName, "_")
local newMapping = "replacement_" .. toNumber(replNumber - 1) local newMapping = "replacement_" .. toNumber(replNumber - 1)
self.dialog:SelectGroup(self.name, "replacements", newMapping) self.dialog:SelectGroup(self.name, "replacements", newMapping)
end end
function Grichelde:DeleteAllReplacements() function Grichelde:DeleteAllMappings()
self:DebugPrint("DeleteAllReplacements : ") self:DebugPrint("DeleteAllMappings")
--self.db.profile.replacements = {} -- do NOT set self.db.profile.replacements = {} it will break defaults
for replName, _ in pairs(self.db.profile.replacements or {}) do for replName, _ in pairs(self.db.profile.replacements or {}) do
self.db.profile.replacements[replName] = nil self.db.profile.replacements[replName] = nil
end end
self:AddReplacement() self:AddEmptyMapping()
self:RefreshOptions("DeleteAllReplacements") self:RefreshOptions("DeleteAllMappings")
end end
function Grichelde:ToogleMinimapButton(info, val)
self:TracePrint("ToogleMinimapButton : info")
for i = 0, #info do
self:TracePrint("%d = %s", i, info[i])
end
self.db.profile.minimapButton.hide = not val
self:DebugPrint("ToogleMinimapButton : hidden: ", self.db.profile.minimapButton.hide)
if self.db.profile.minimapButton.hide then
self.icon:Hide(self.name);
else
self.icon:Show(self.name);
end
end

89
GricheldeUpgrade.lua Normal file
View File

@@ -0,0 +1,89 @@
-- read namespace from global env
local _G = _G
local Grichelde = _G.Grichelde
local pairs, find, toNumber = Grichelde.functions.pairs, Grichelde.functions.find, Grichelde.functions.toNumber
function Grichelde:Upgrade_To_v060()
self:PrefixedPrint(self.L.Upgrade_ToVersion, Grichelde.COLOR_CODES.ORANGE .. "0.6.0" .. Grichelde.COLOR_CODES.CLOSE)
local replacements = self.db.profile.replacements or {}
self:DebugPrint("Upgrade_To_v060 : old database")
self:DebugPrint(replacements)
for _, replTable in pairs(replacements) do
replTable["ignoreCase"] = not replTable["caseSensitive"]
replTable["caseSensitive"] = nil
end
self:DebugPrint("Upgrade_To_v060 : new database")
self:DebugPrint(replacements)
return 0, 6, 0
end
function Grichelde:Upgrade_To_v070()
self:PrefixedPrint(self.L.Upgrade_ToVersion, Grichelde.COLOR_CODES.ORANGE .. "0.7.0" .. Grichelde.COLOR_CODES.CLOSE)
local replacements = self.db.profile.replacements or {}
self:DebugPrint("Upgrade_To_v070 : old database")
self:DebugPrint(replacements)
for _, replTable in pairs(replacements) do
replTable["exactCase"] = not replTable["ignoreCase"]
replTable["ignoreCase"] = nil
end
self:DebugPrint("Upgrade_To_v070 : new database")
self:DebugPrint(replacements)
return 0, 7, 0
end
function Grichelde:Upgrade_To_v072()
self:PrefixedPrint(self.L.Upgrade_ToVersion, Grichelde.COLOR_CODES.ORANGE .. "0.7.2" .. Grichelde.COLOR_CODES.CLOSE)
self:DebugPrint("Upgrade_To_v072 : old database")
self:DebugPrint(self.db.profile)
self.db.profile["minimapButton"] = { hide = false }
self:DebugPrint("Upgrade_To_v072 : new database")
self:DebugPrint(self.db.profile)
return 0, 7, 2
end
function Grichelde:UpgradeDatabase()
local dbVersion = self.db.global.version or "0.0.0"
self:DebugPrint("Database version:", dbVersion)
local _, _, maj, min, pat = find(dbVersion, "(%d+)%.(%d+)%.(%d+).*")
local major, minor, patch = toNumber(maj) or 0, toNumber(min) or 0, toNumber(pat) or 0
local upgrade = 0
local error = false
if major == 0 then
if minor < 6 then
upgrade = upgrade + 1
major, minor, patch = self:Upgrade_To_v060(dbVersion)
end
if minor < 7 then
upgrade = upgrade + 1
major, minor, patch = self:Upgrade_To_v070(dbVersion)
end
if minor == 7 then
if patch < 2 then
upgrade = upgrade + 1
major, minor, patch = self:Upgrade_To_v072(dbVersion)
end
end
end
if upgrade == 0 then
self:DebugPrint("Database up-to-date")
else
if not error then
self.db.global.version = self.version
self:PrefixedPrint(Grichelde.COLOR_CODES.GREEN .. self.L.Upgrade_Successful .. Grichelde.COLOR_CODES.CLOSE)
end
end
end

View File

@@ -65,9 +65,9 @@ function Grichelde:SplitOnFirstMatch(text, delimPattern, start)
if text == nil then return nil end if text == nil then return nil end
local pattern = "^(.-)" .. (delimPattern or " " ) .."(.*)" local pattern = "^(.-)" .. (delimPattern or " " ) .."(.*)"
local pos = start or 1 local pos = start or 1
self:DebugPrint("SplitOnFirstMatch : text: %s, pattern: %s, start: %d", text, pattern, start) self:TracePrint("SplitOnFirstMatch : text: %s, pattern: %s, start: %d", text, pattern, start)
local _, _, left, right = find(text, pattern, pos) local _, _, left, right = find(text, pattern, pos)
self:DebugPrint("SplitOnFirstMatch : left: %s, right: %s", left, right) self:TracePrint("SplitOnFirstMatch : left: %s, right: %s", left, right)
return left or text, right return left or text, right
end end
@@ -75,9 +75,9 @@ end
function Grichelde:SplitOnLastMatch(text, delimPattern, start) function Grichelde:SplitOnLastMatch(text, delimPattern, start)
local pattern = "(.*)" .. (delimPattern or " ") .. "(.-)$" local pattern = "(.*)" .. (delimPattern or " ") .. "(.-)$"
local pos = start or 1 local pos = start or 1
self:DebugPrint("SplitOnLastMatch : text: %s, pattern: %s, start: %d", text, pattern, start) self:TracePrint("SplitOnLastMatch : text: %s, pattern: %s, start: %d", text, pattern, start)
local _, _, left, right = find(text, pattern, pos) local _, _, left, right = find(text, pattern, pos)
self:DebugPrint("SplitOnLastMatch : left: %s, right: %s", left, right) self:TracePrint("SplitOnLastMatch : left: %s, right: %s", left, right)
return left, right or text return left, right or text
end end
@@ -121,18 +121,27 @@ function Grichelde:PrefixedPrint(...)
end end
function Grichelde:DebugPrint(obj, ...) function Grichelde:DebugPrint(obj, ...)
local function prefixedDebugPrint(...) self:LogPrint(Grichelde.LOG_LEVEL.DEBUG, function(...)
print(self.COLOR_CODES.GRAY .. self.L.AddonName .. self.COLOR_CODES.CLOSE .. ":", self:Format(...)) print(self.COLOR_CODES.GRAY .. self.L.AddonName .. self.COLOR_CODES.CLOSE .. ":", self:Format(...))
end, obj, ...)
end end
if (self.debug) then function Grichelde:TracePrint(obj, ...)
self:LogPrint(Grichelde.LOG_LEVEL.TRACE, function(...)
print(self.COLOR_CODES.DARKGRAY .. self.L.AddonName .. self.COLOR_CODES.CLOSE .. ":", self:Format(...))
end, obj, ...)
end
function Grichelde:LogPrint(logLevel, printFunc, obj, ...)
if (self.logLevel >= logLevel) then
local printF = printFunc or print
if obj == nil then if obj == nil then
prefixedDebugPrint("<nil>") printF("<nil>")
else else
if type(obj) == "string" then if type(obj) == "string" then
local l = select("#", ...) local l = select("#", ...)
if ( l == 0 or not find(obj, "%%")) then if ( l == 0 or not find(obj, "%%")) then
prefixedDebugPrint(obj, ...) printF(obj, ...)
else else
-- sanitize nil values in vararg -- sanitize nil values in vararg
local packed = { ... } local packed = { ... }
@@ -144,13 +153,91 @@ function Grichelde:DebugPrint(obj, ...)
-- self:tPrint(packed) -- self:tPrint(packed)
-- cannot assign unpacked to a vararg variable and print it for debug -- cannot assign unpacked to a vararg variable and print it for debug
local fmtMsg = format(obj, unpack(packed, 1, l)) -- manually set count as unpack() stops on nil (bug with #table) local fmtMsg = format(obj, unpack(packed, 1, l)) -- manually set count as unpack() stops on nil (bug with #table)
prefixedDebugPrint(fmtMsg) printF(fmtMsg)
end end
elseif type(obj) == "table" then elseif type(obj) == "table" then
tPrint(obj, 0, {}, prefixedDebugPrint) tPrint(obj, 0, {}, printF)
else else
prefixedDebugPrint(plainValue(obj)) printF(plainValue(obj))
end end
end end
end end
end end
--- Print UI options to chat frame
function Grichelde:PrintOptions()
self:PrefixedPrint(self.COLOR_CODES.PREFIX .. self.L.Debug_Options .. self.COLOR_CODES.CLOSE)
self:LogPrint(-1, function(...)
print(self.COLOR_CODES.PREFIX .. self.L.AddonName .. self.COLOR_CODES.CLOSE .. ":", self:Format(...))
end, self.options.args.replacements.args)
end
--- Print DB
function Grichelde:PrintProfile()
self:PrefixedPrint(self.COLOR_CODES.PREFIX .. self.L.Debug_Profile .. self.COLOR_CODES.CLOSE)
self:LogPrint(-1, function(...)
print(self.COLOR_CODES.PREFIX .. self.L.AddonName .. self.COLOR_CODES.CLOSE .. ":", self:Format(...))
end, self.db.profile)
end
--- Print DB replacements to chat frame
function Grichelde:PrintMappings()
self:PrefixedPrint(self.COLOR_CODES.PREFIX .. self.L.Debug_Mappings .. self.COLOR_CODES.CLOSE)
self:LogPrint(-1, function(...)
print(self.COLOR_CODES.PREFIX .. self.L.AddonName .. self.COLOR_CODES.CLOSE .. ":", self:Format(...))
end, self.db.profile.replacements)
end
--- Open UI windows with replacements in it
function Grichelde:ToogleMappings()
local AceGUI = LibStub("AceGUI-3.0")
if self.debugFrame then
AceGUI:Release(self.debugFrame)
self.debugFrame = nil
else
local replacements = self.db.profile.replacements or {}
local repls = 0
for k, _ in pairs(replacements) do
if k and find(k, "^replacement_") then
repls = repls + 1
end
end
local frame = AceGUI:Create("Frame");
frame:SetTitle(self.L.Debug_Mappings);
frame:SetStatusText(self:Format(self.L.Debug_Mappings_Found, repls))
frame:SetWidth(400);
frame:SetAutoAdjustHeight(true)
--frame:SetFullHeight(true)
frame:SetLayout("Flow");
local function closeFrame(widget) AceGUI:Release(widget); self.debugFrame = nil end
frame:SetCallback("OnClose", closeFrame)
local hint = AceGUI:Create("Label");
hint:SetText(self.L.Debug_Mappings_Hint)
hint:SetFullWidth(true)
frame:AddChild(hint);
local scroll = AceGUI:Create("ScrollFrame");
scroll:SetFullWidth(true)
scroll:SetFullHeight(true)
scroll:SetLayout("Fill");
local configBox = AceGUI:Create("MultiLineEditBox");
configBox:SetLabel("")
local text = ""
tPrint(replacements, 0, {}, function(s) text = text .. s .. "|n" end)
configBox:SetText(text)
configBox:SetFullWidth(true)
configBox:SetFullHeight(true)
configBox:DisableButton(true)
--configBox:SetDisabled(true)
configBox:SetNumLines(50);
configBox:SetFocus()
scroll:AddChild(configBox);
frame:AddChild(scroll);
self.debugFrame = frame
end
end

View File

@@ -0,0 +1,470 @@
-----------------------------------------------------------------------
-- LibDBIcon-1.0
--
-- Allows addons to easily create a lightweight minimap icon as an alternative to heavier LDB displays.
--
local DBICON10 = "LibDBIcon-1.0"
local DBICON10_MINOR = 43 -- Bump on changes
if not LibStub then error(DBICON10 .. " requires LibStub.") end
local ldb = LibStub("LibDataBroker-1.1", true)
if not ldb then error(DBICON10 .. " requires LibDataBroker-1.1.") end
local lib = LibStub:NewLibrary(DBICON10, DBICON10_MINOR)
if not lib then return end
lib.objects = lib.objects or {}
lib.callbackRegistered = lib.callbackRegistered or nil
lib.callbacks = lib.callbacks or LibStub("CallbackHandler-1.0"):New(lib)
lib.notCreated = lib.notCreated or {}
lib.radius = lib.radius or 5
lib.tooltip = lib.tooltip or CreateFrame("GameTooltip", "LibDBIconTooltip", UIParent, "GameTooltipTemplate")
local next, Minimap = next, Minimap
local isDraggingButton = false
function lib:IconCallback(event, name, key, value)
if lib.objects[name] then
if key == "icon" then
lib.objects[name].icon:SetTexture(value)
elseif key == "iconCoords" then
lib.objects[name].icon:UpdateCoord()
elseif key == "iconR" then
local _, g, b = lib.objects[name].icon:GetVertexColor()
lib.objects[name].icon:SetVertexColor(value, g, b)
elseif key == "iconG" then
local r, _, b = lib.objects[name].icon:GetVertexColor()
lib.objects[name].icon:SetVertexColor(r, value, b)
elseif key == "iconB" then
local r, g = lib.objects[name].icon:GetVertexColor()
lib.objects[name].icon:SetVertexColor(r, g, value)
end
end
end
if not lib.callbackRegistered then
ldb.RegisterCallback(lib, "LibDataBroker_AttributeChanged__icon", "IconCallback")
ldb.RegisterCallback(lib, "LibDataBroker_AttributeChanged__iconCoords", "IconCallback")
ldb.RegisterCallback(lib, "LibDataBroker_AttributeChanged__iconR", "IconCallback")
ldb.RegisterCallback(lib, "LibDataBroker_AttributeChanged__iconG", "IconCallback")
ldb.RegisterCallback(lib, "LibDataBroker_AttributeChanged__iconB", "IconCallback")
lib.callbackRegistered = true
end
local function getAnchors(frame)
local x, y = frame:GetCenter()
if not x or not y then return "CENTER" 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
local function onEnter(self)
if isDraggingButton then return end
for _, button in next, lib.objects do
if button.showOnMouseover then
button.fadeOut:Stop()
button:SetAlpha(1)
end
end
local obj = self.dataObject
if obj.OnTooltipShow then
lib.tooltip:SetOwner(self, "ANCHOR_NONE")
lib.tooltip:SetPoint(getAnchors(self))
obj.OnTooltipShow(lib.tooltip)
lib.tooltip:Show()
elseif obj.OnEnter then
obj.OnEnter(self)
end
end
local function onLeave(self)
lib.tooltip:Hide()
if not isDraggingButton then
for _, button in next, lib.objects do
if button.showOnMouseover then
button.fadeOut:Play()
end
end
end
local obj = self.dataObject
if obj.OnLeave then
obj.OnLeave(self)
end
end
--------------------------------------------------------------------------------
local onDragStart, updatePosition
do
local minimapShapes = {
["ROUND"] = {true, true, true, true},
["SQUARE"] = {false, false, false, false},
["CORNER-TOPLEFT"] = {false, false, false, true},
["CORNER-TOPRIGHT"] = {false, false, true, false},
["CORNER-BOTTOMLEFT"] = {false, true, false, false},
["CORNER-BOTTOMRIGHT"] = {true, false, false, false},
["SIDE-LEFT"] = {false, true, false, true},
["SIDE-RIGHT"] = {true, false, true, false},
["SIDE-TOP"] = {false, false, true, true},
["SIDE-BOTTOM"] = {true, true, false, false},
["TRICORNER-TOPLEFT"] = {false, true, true, true},
["TRICORNER-TOPRIGHT"] = {true, false, true, true},
["TRICORNER-BOTTOMLEFT"] = {true, true, false, true},
["TRICORNER-BOTTOMRIGHT"] = {true, true, true, false},
}
local rad, cos, sin, sqrt, max, min = math.rad, math.cos, math.sin, math.sqrt, math.max, math.min
function updatePosition(button, position)
local angle = rad(position or 225)
local x, y, q = cos(angle), sin(angle), 1
if x < 0 then q = q + 1 end
if y > 0 then q = q + 2 end
local minimapShape = GetMinimapShape and GetMinimapShape() or "ROUND"
local quadTable = minimapShapes[minimapShape]
local w = (Minimap:GetWidth() / 2) + lib.radius
local h = (Minimap:GetHeight() / 2) + lib.radius
if quadTable[q] then
x, y = x*w, y*h
else
local diagRadiusW = sqrt(2*(w)^2)-10
local diagRadiusH = sqrt(2*(h)^2)-10
x = max(-w, min(x*diagRadiusW, w))
y = max(-h, min(y*diagRadiusH, h))
end
button:SetPoint("CENTER", Minimap, "CENTER", x, y)
end
end
local function onClick(self, b)
if self.dataObject.OnClick then
self.dataObject.OnClick(self, b)
end
end
local function onMouseDown(self)
self.isMouseDown = true
self.icon:UpdateCoord()
end
local function onMouseUp(self)
self.isMouseDown = false
self.icon:UpdateCoord()
end
do
local deg, atan2 = math.deg, math.atan2
local function onUpdate(self)
local mx, my = Minimap:GetCenter()
local px, py = GetCursorPosition()
local scale = Minimap:GetEffectiveScale()
px, py = px / scale, py / scale
local pos = 225
if self.db then
pos = deg(atan2(py - my, px - mx)) % 360
self.db.minimapPos = pos
else
pos = deg(atan2(py - my, px - mx)) % 360
self.minimapPos = pos
end
updatePosition(self, pos)
end
function onDragStart(self)
self:LockHighlight()
self.isMouseDown = true
self.icon:UpdateCoord()
self:SetScript("OnUpdate", onUpdate)
isDraggingButton = true
lib.tooltip:Hide()
for _, button in next, lib.objects do
if button.showOnMouseover then
button.fadeOut:Stop()
button:SetAlpha(1)
end
end
end
end
local function onDragStop(self)
self:SetScript("OnUpdate", nil)
self.isMouseDown = false
self.icon:UpdateCoord()
self:UnlockHighlight()
isDraggingButton = false
for _, button in next, lib.objects do
if button.showOnMouseover then
button.fadeOut:Play()
end
end
end
local defaultCoords = {0, 1, 0, 1}
local function updateCoord(self)
local coords = self:GetParent().dataObject.iconCoords or defaultCoords
local deltaX, deltaY = 0, 0
if not self:GetParent().isMouseDown then
deltaX = (coords[2] - coords[1]) * 0.05
deltaY = (coords[4] - coords[3]) * 0.05
end
self:SetTexCoord(coords[1] + deltaX, coords[2] - deltaX, coords[3] + deltaY, coords[4] - deltaY)
end
local function createButton(name, object, db)
local button = CreateFrame("Button", "LibDBIcon10_"..name, Minimap)
button.dataObject = object
button.db = db
button:SetFrameStrata("MEDIUM")
button:SetSize(31, 31)
button:SetFrameLevel(8)
button:RegisterForClicks("anyUp")
button:RegisterForDrag("LeftButton")
button:SetHighlightTexture(136477) --"Interface\\Minimap\\UI-Minimap-ZoomButton-Highlight"
local overlay = button:CreateTexture(nil, "OVERLAY")
overlay:SetSize(53, 53)
overlay:SetTexture(136430) --"Interface\\Minimap\\MiniMap-TrackingBorder"
overlay:SetPoint("TOPLEFT")
local background = button:CreateTexture(nil, "BACKGROUND")
background:SetSize(20, 20)
background:SetTexture(136467) --"Interface\\Minimap\\UI-Minimap-Background"
background:SetPoint("TOPLEFT", 7, -5)
local icon = button:CreateTexture(nil, "ARTWORK")
icon:SetSize(17, 17)
icon:SetTexture(object.icon)
icon:SetPoint("TOPLEFT", 7, -6)
button.icon = icon
button.isMouseDown = false
local r, g, b = icon:GetVertexColor()
icon:SetVertexColor(object.iconR or r, object.iconG or g, object.iconB or b)
icon.UpdateCoord = updateCoord
icon:UpdateCoord()
button:SetScript("OnEnter", onEnter)
button:SetScript("OnLeave", onLeave)
button:SetScript("OnClick", onClick)
if not db or not db.lock then
button:SetScript("OnDragStart", onDragStart)
button:SetScript("OnDragStop", onDragStop)
end
button:SetScript("OnMouseDown", onMouseDown)
button:SetScript("OnMouseUp", onMouseUp)
button.fadeOut = button:CreateAnimationGroup()
local animOut = button.fadeOut:CreateAnimation("Alpha")
animOut:SetOrder(1)
animOut:SetDuration(0.2)
animOut:SetFromAlpha(1)
animOut:SetToAlpha(0)
animOut:SetStartDelay(1)
button.fadeOut:SetToFinalAlpha(true)
lib.objects[name] = button
if lib.loggedIn then
updatePosition(button, db and db.minimapPos)
if not db or not db.hide then
button:Show()
else
button:Hide()
end
end
lib.callbacks:Fire("LibDBIcon_IconCreated", button, name) -- Fire 'Icon Created' callback
end
-- We could use a metatable.__index on lib.objects, but then we'd create
-- the icons when checking things like :IsRegistered, which is not necessary.
local function check(name)
if lib.notCreated[name] then
createButton(name, lib.notCreated[name][1], lib.notCreated[name][2])
lib.notCreated[name] = nil
end
end
-- Wait a bit with the initial positioning to let any GetMinimapShape addons
-- load up.
if not lib.loggedIn then
local f = CreateFrame("Frame")
f:SetScript("OnEvent", function(f)
for _, button in next, lib.objects do
updatePosition(button, button.db and button.db.minimapPos)
if not button.db or not button.db.hide then
button:Show()
else
button:Hide()
end
end
lib.loggedIn = true
f:SetScript("OnEvent", nil)
end)
f:RegisterEvent("PLAYER_LOGIN")
end
local function getDatabase(name)
return lib.notCreated[name] and lib.notCreated[name][2] or lib.objects[name].db
end
function lib:Register(name, object, db)
if not object.icon then error("Can't register LDB objects without icons set!") end
if lib.objects[name] or lib.notCreated[name] then error(DBICON10.. ": Object '".. name .."' is already registered.") end
if not db or not db.hide then
createButton(name, object, db)
else
lib.notCreated[name] = {object, db}
end
end
function lib:Lock(name)
if not lib:IsRegistered(name) then return end
if lib.objects[name] then
lib.objects[name]:SetScript("OnDragStart", nil)
lib.objects[name]:SetScript("OnDragStop", nil)
end
local db = getDatabase(name)
if db then
db.lock = true
end
end
function lib:Unlock(name)
if not lib:IsRegistered(name) then return end
if lib.objects[name] then
lib.objects[name]:SetScript("OnDragStart", onDragStart)
lib.objects[name]:SetScript("OnDragStop", onDragStop)
end
local db = getDatabase(name)
if db then
db.lock = nil
end
end
function lib:Hide(name)
if not lib.objects[name] then return end
lib.objects[name]:Hide()
end
function lib:Show(name)
check(name)
local button = lib.objects[name]
if button then
button:Show()
updatePosition(button, button.db and button.db.minimapPos or button.minimapPos)
end
end
function lib:IsRegistered(name)
return (lib.objects[name] or lib.notCreated[name]) and true or false
end
function lib:Refresh(name, db)
check(name)
local button = lib.objects[name]
if db then
button.db = db
end
updatePosition(button, button.db and button.db.minimapPos or button.minimapPos)
if not button.db or not button.db.hide then
button:Show()
else
button:Hide()
end
if not button.db or not button.db.lock then
button:SetScript("OnDragStart", onDragStart)
button:SetScript("OnDragStop", onDragStop)
else
button:SetScript("OnDragStart", nil)
button:SetScript("OnDragStop", nil)
end
end
function lib:GetMinimapButton(name)
return lib.objects[name]
end
do
local function OnMinimapEnter()
if isDraggingButton then return end
for _, button in next, lib.objects do
if button.showOnMouseover then
button.fadeOut:Stop()
button:SetAlpha(1)
end
end
end
local function OnMinimapLeave()
if isDraggingButton then return end
for _, button in next, lib.objects do
if button.showOnMouseover then
button.fadeOut:Play()
end
end
end
Minimap:HookScript("OnEnter", OnMinimapEnter)
Minimap:HookScript("OnLeave", OnMinimapLeave)
function lib:ShowOnEnter(name, value)
local button = lib.objects[name]
if button then
if value then
button.showOnMouseover = true
button.fadeOut:Stop()
button:SetAlpha(0)
else
button.showOnMouseover = false
button.fadeOut:Stop()
button:SetAlpha(1)
end
end
end
end
function lib:GetButtonList()
local t = {}
for name in next, lib.objects do
t[#t+1] = name
end
return t
end
function lib:SetButtonRadius(radius)
if type(radius) == "number" then
lib.radius = radius
for _, button in next, lib.objects do
updatePosition(button, button.db and button.db.minimapPos or button.minimapPos)
end
end
end
function lib:SetButtonToPosition(button, position)
updatePosition(lib.objects[button] or button, position)
end
-- Upgrade!
for name, button in next, lib.objects do
local db = getDatabase(name)
if not db or not db.lock then
button:SetScript("OnDragStart", onDragStart)
button:SetScript("OnDragStop", onDragStop)
end
button:SetScript("OnEnter", onEnter)
button:SetScript("OnLeave", onLeave)
button:SetScript("OnClick", onClick)
button:SetScript("OnMouseDown", onMouseDown)
button:SetScript("OnMouseUp", onMouseUp)
if not button.fadeOut then -- Upgrade to 39
button.fadeOut = button:CreateAnimationGroup()
local animOut = button.fadeOut:CreateAnimation("Alpha")
animOut:SetOrder(1)
animOut:SetDuration(0.2)
animOut:SetFromAlpha(1)
animOut:SetToAlpha(0)
animOut:SetStartDelay(1)
button.fadeOut:SetToFinalAlpha(true)
end
end
lib:SetButtonRadius(lib.radius) -- Upgrade to 40

View File

@@ -0,0 +1,7 @@
<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="LibDBIcon-1.0.lua"/>
</Ui>

View File

@@ -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

View File

@@ -0,0 +1,13 @@
LibDataBroker is a small WoW addon library designed to provide a "MVC":http://en.wikipedia.org/wiki/Model-view-controller interface for use in various addons.
LDB's primary goal is to "detach" plugins for TitanPanel and FuBar from the display addon.
Plugins can provide data into a simple table, and display addons can receive callbacks to refresh their display of this data.
LDB also provides a place for addons to register "quicklaunch" functions, removing the need for authors to embed many large libraries to create minimap buttons.
Users who do not wish to be "plagued" by these buttons simply do not install an addon to render them.
Due to it's simple generic design, LDB can be used for any design where you wish to have an addon notified of changes to a table.
h2. Links
* "API documentation":http://github.com/tekkub/libdatabroker-1-1/wikis/api
* "Data specifications":http://github.com/tekkub/libdatabroker-1-1/wikis/data-specifications
* "Addons using LDB":http://github.com/tekkub/libdatabroker-1-1/wikis/addons-using-ldb

View File

@@ -8,4 +8,27 @@ Intentionally started as a helper addon for a roleplaying friend, Grichelde can
* write out abbreviations for you * write out abbreviations for you
* create hilarious moments during roleplay * create hilarious moments during roleplay
<!-- Did you ever had an Undead without a jaw and wondered how it might sound like if s/he'd actually speak? --> ## FAQ
### How do I start
Grichelde is active right from the start with default mappings. To open the options UI, either left-click on
the new minimap icon, or type `/gri` or `/grichelde` in the chatbox. All mappings and channels can be configured there.
### My replacement is not taken.
After entering a search or replacement text, you see a button "Okay" next to yout input. This is **not** a validation message,
but the save button for text. This is a rectriction from the UI library and can be seen in other addons as well.
Please click on "Okay" button to save the input permanently.
### I get errors, what should I do?
Please report your errors here. Make a screenshot or copy both the error message as well as your recent mappings.
You can bring up a small windows with your mapping by right-clicking the minimap icon or entering the "/gri mappings" command.
### Why that strange name?
Grichelde is the name of the undead rogue without a jaw, that was played in RP session with my friend.
She started replacing "s" and "t" letters manually for each line in the chat, which is cumbersome over time.
(If you wondered how an Undead without a jaw sounds like, its really hilarious.) Without spelling errors,
"Griselde" in German would be a old-fashioned female first name.

View File

@@ -2,14 +2,16 @@
..\FrameXML\UI.xsd"> ..\FrameXML\UI.xsd">
<Script file="Libs\LibStub\LibStub.lua"/> <Script file="Libs\LibStub\LibStub.lua"/>
<Script file="Libs\LibDataBroker-1.1\LibDataBroker-1.1.lua" />
<Script file="Libs\LibDBIcon-1.0\LibDBIcon-1.0.lua" />
<Include file="Libs\CallbackHandler-1.0\CallbackHandler-1.0.xml"/> <Include file="Libs\CallbackHandler-1.0\CallbackHandler-1.0.xml"/>
<Include file="Libs\AceAddon-3.0\AceAddon-3.0.xml"/> <Include file="Libs\AceAddon-3.0\AceAddon-3.0.xml"/>
<Include file="Libs\AceConfig-3.0\AceConfig-3.0.xml"/> <Include file="Libs\AceConfig-3.0\AceConfig-3.0.xml"/>
<Include file="Libs\AceConsole-3.0\AceConsole-3.0.xml"/> <Include file="Libs\AceConsole-3.0\AceConsole-3.0.xml"/>
<Include file="Libs\AceDB-3.0\AceDB-3.0.xml"/> <Include file="Libs\AceDB-3.0\AceDB-3.0.xml"/>
<Include file="Libs\AceDBOptions-3.0\AceDBOptions-3.0.xml"/> <Include file="Libs\AceDBOptions-3.0\AceDBOptions-3.0.xml"/>
<Include file="Libs\AceGUI-3.0\AceGUI-3.0.xml"/>
<Include file="Libs\AceEvent-3.0\AceEvent-3.0.xml" /> <Include file="Libs\AceEvent-3.0\AceEvent-3.0.xml" />
<Include file="Libs\AceLocale-3.0\AceLocale-3.0.xml" /> <Include file="Libs\AceGUI-3.0\AceGUI-3.0.xml"/>
<Include file="Libs\AceHook-3.0\AceHook-3.0.xml"/> <Include file="Libs\AceHook-3.0\AceHook-3.0.xml"/>
<Include file="Libs\AceLocale-3.0\AceLocale-3.0.xml" />
</Ui> </Ui>

View File

@@ -5,8 +5,13 @@ if not L then return end
L.AddonName = "Grichelde" L.AddonName = "Grichelde"
L.VersionAbbr = "v" L.VersionAbbr = "v"
L.AddonLoaded = "%s hilft Euch jetzt bei euren Sprachschwierigkeiten." L.AddonLoaded = "%s hilft Euch jetzt bei euren Sprachschwierigkeiten."
L.Addon_Detected_Misspelled = "Das Addon 'Misspelled' wurde erkannt und alle Nachrichten werden automatisch bereinigt." L.Upgrade_ToVersion = "Hebe Databank auf Version %s an."
L.Addon_Detected_WIM = "Das Addon 'WIM' wurde erkannt und alle Flüsternnachrichten aus IM-Fenster werden behandelt." L.Upgrade_Successful = "Upgrade erfolgreich."
L.Debug_Options = "Optionen"
L.Debug_Mappings = "Ersetzungen"
L.Debug_Mappings_Hint = "Der Inhalt der Textbox dient nur zur Fehlersuche und kann herauskopiert werden. Es werden keine Werte aus dieser Textbox eingelesen oder anderweitig verwertet."
L.Debug_Mappings_Found = "%d Ersetzungen gefunden"
L.Debug_Profile = "Profil"
-- profiles -- profiles
L.Profiles_Available = "Verf\195\188gbare Profile:" L.Profiles_Available = "Verf\195\188gbare Profile:"
@@ -23,35 +28,43 @@ L.Profiles_DeleteError = "Das aktive Profil kann nicht gel\195\182scht werden!"
L.Options_Title = "%s Einstellungen" L.Options_Title = "%s Einstellungen"
L.Options_Enabled_Name = "Aktiv" L.Options_Enabled_Name = "Aktiv"
L.Options_Enabled_Desc = "Aktiviert %s" L.Options_Enabled_Desc = "Aktiviert %s"
L.Options_Minimap_Button_Name = "Zeige Minimap-Knopf"
L.Options_Minimap_Button_Desc = "Zeigt oder versteckt den Knopf an der Miniaturkarte"
L.Options_Minimap_Tooltip_Options_Left = "Linksklick"
L.Options_Minimap_Tooltip_Options_Right = "\195\150ffnet oder schlie\195\159t die Einstellungen."
L.Options_Minimap_Tooltip_Mappings_Left = "Rechtsklick"
L.Options_Minimap_Tooltip_Mappings_Right = "\195\150ffnet das Debug-Fenster mit Ersetzungscode."
L.Options_Channels_Group_Name = "Kan\195\164le" L.Options_Channels_Group_Name = "Kan\195\164le"
L.Options_Channels_Group_Desc = "%s ist in folgenden Kan\195\164len aktiv." L.Options_Channels_Group_Desc = "%s ist in folgenden Kan\195\164len aktiv."
L.Options_Channels_ChannelSay_Name = "Sagen" L.Options_Channels_Header = "Eine Ersetzung wird nur in den unten markierten Kan\195\164len durchgef\195\188hrt:"
L.Options_Channels_ChannelSay_Desc = "Aktiviert %s im Kanal \"Sagen\"."
L.Options_Channels_ChannelEmote_Name = "Emote" L.Options_Channel_Say_Name = "Sagen"
L.Options_Channels_ChannelEmote_Desc = "Aktiviert %s im Kanal \"Emote\"." L.Options_Channel_Say_Desc = "Aktiviert %s im Kanal \"Sagen\"."
L.Options_Channels_ChannelYell_Name = "Schreien" L.Options_Channel_Emote_Name = "Emote"
L.Options_Channels_ChannelYell_Desc = "Aktiviert %s im Kanal \"Schreien\"." L.Options_Channel_Emote_Desc = "Aktiviert %s im Kanal \"Emote\"."
L.Options_Channels_ChannelParty_Name = "Gruppe" L.Options_Channel_Yell_Name = "Schreien"
L.Options_Channels_ChannelParty_Desc = "Aktiviert %s im Kanal \"Gruppe\"." L.Options_Channel_Yell_Desc = "Aktiviert %s im Kanal \"Schreien\"."
L.Options_Channels_ChannelPartyLeader_Name = "Gruppenanf\195\188hrer" L.Options_Channel_Party_Name = "Gruppe"
L.Options_Channels_ChannelPartyLeader_Desc = "Aktiviert %s im Kanal \"Gruppenanf\195\188hrer\"." L.Options_Channel_Party_Desc = "Aktiviert %s im Kanal \"Gruppe\"."
L.Options_Channels_ChannelGuild_Name = "Gilde" L.Options_Channel_PartyLeader_Name = "Gruppenanf\195\188hrer"
L.Options_Channels_ChannelGuild_Desc = "Aktiviert %s im Kanal \"Gilde\"." L.Options_Channel_PartyLeader_Desc = "Aktiviert %s im Kanal \"Gruppenanf\195\188hrer\"."
L.Options_Channels_ChannelOfficer_Name = "Offiziere" L.Options_Channel_Guild_Name = "Gilde"
L.Options_Channels_ChannelOfficer_Desc = "Aktiviert %s im Kanal \"Offiziere\"." L.Options_Channel_Guild_Desc = "Aktiviert %s im Kanal \"Gilde\"."
L.Options_Channels_ChannelRaid_Name = "Schlachtzug" L.Options_Channel_Officer_Name = "Offiziere"
L.Options_Channels_ChannelRaid_Desc = "Aktiviert %s im Kanal \"Schlachtzug\"." L.Options_Channel_Officer_Desc = "Aktiviert %s im Kanal \"Offiziere\"."
L.Options_Channels_ChannelRaidLeader_Name = "Schlachtzugsanf\195\188hrer" L.Options_Channel_Raid_Name = "Schlachtzug"
L.Options_Channels_ChannelRaidLeader_Desc = "Aktiviert %s im Kanal \"Schlachtzugsanf\195\188hrer\"." L.Options_Channel_Raid_Desc = "Aktiviert %s im Kanal \"Schlachtzug\"."
L.Options_Channels_ChannelRaidWarning_Name = "Schlachtzugswarnung" L.Options_Channel_RaidLeader_Name = "Schlachtzugsanf\195\188hrer"
L.Options_Channels_ChannelRaidWarning_Desc = "Aktiviert %s im Kanal \"Schlachtzugswarnung." L.Options_Channel_RaidLeader_Desc = "Aktiviert %s im Kanal \"Schlachtzugsanf\195\188hrer\"."
L.Options_Channels_ChannelInstance_Name = "Instanz" L.Options_Channel_RaidWarning_Name = "Schlachtzugswarnung"
L.Options_Channels_ChannelInstance_Desc = "Aktiviert %s im Kanal \"Instanz\"." L.Options_Channel_RaidWarning_Desc = "Aktiviert %s im Kanal \"Schlachtzugswarnung."
L.Options_Channels_ChannelBattleground_Name = "Schlachtfeld" L.Options_Channel_Instance_Name = "Instanz"
L.Options_Channels_ChannelBattleground_Desc = "Aktiviert %s im Kanal \"Schlachtfeld\"." L.Options_Channel_Instance_Desc = "Aktiviert %s im Kanal \"Instanz\"."
L.Options_Channels_ChannelWhisper_Name = "Fl\195\188stern" L.Options_Channel_Battleground_Name = "Schlachtfeld"
L.Options_Channels_ChannelWhisper_Desc = "Aktiviert %s im Kanal \"Fl\195\188stern\"." L.Options_Channel_Battleground_Desc = "Aktiviert %s im Kanal \"Schlachtfeld\"."
L.Options_Channel_Whisper_Name = "Fl\195\188stern"
L.Options_Channel_Whisper_Desc = "Aktiviert %s im Kanal \"Fl\195\188stern\"."
L.Options_Replacements_Group_Name = "Ersetzungen" L.Options_Replacements_Group_Name = "Ersetzungen"
L.Options_Replacements_Group_Desc = "Diese Vorkommen werden in den aktivierten Kan\195\164len ersetzt." L.Options_Replacements_Group_Desc = "Diese Vorkommen werden in den aktivierten Kan\195\164len ersetzt."
@@ -60,18 +73,25 @@ L.Options_Replacements_Add_Desc = "F\195\188gt eine neue Zuordnung hinzu."
L.Options_Replacements_DeleteAll_Name = "Alle L\195\182schen" L.Options_Replacements_DeleteAll_Name = "Alle L\195\182schen"
L.Options_Replacements_DeleteAll_Desc = "L\195\182scht alle Zuweisungen." L.Options_Replacements_DeleteAll_Desc = "L\195\182scht alle Zuweisungen."
L.Options_Replacements_DeleteAll_ConfirmText="Wirklich ALLE Zuweisungen l\195\182schen?" L.Options_Replacements_DeleteAll_ConfirmText="Wirklich ALLE Zuweisungen l\195\182schen?"
L.Options_Replacements_Header = "Die Vorkommen links vom Pfeil ( => ) werden in den aktivierten Kan\195\164len gesucht und durch den Text rechts vom Pfeil ersetzt."
L.Options_Replacement_Group_Name = "%s => %s" .."|nWird die Gro\195\159- und Kleinschreibung ignoriert, wird die Gro\195\159schreibung jedes Zeichens bei der Ersetzung \195\188bernommen."
L.Options_Replacement_Group_Desc = "Dieses Vorkommen wird in den aktivierten Kan\195\164len ersetzt." .."|nDas Zusammenfassen aufeinanderfolgender Treffer vermeidet unsch\195\182ne Wiederholungen, die durch die Ersetzung entstehen k\195\182nnen."
L.Options_Replacement_EmptyMapping = "(keine)" .."|nMit den beiden Standard-Ersetzung wird so aus \"Tasse\" => \"Ckache\"."
L.Options_Replacement_SearchText_Name = "Suchtext:" L.Options_Mapping_Group_Name = "%s => %s"
L.Options_Replacement_SearchText_Desc = "Dieser Text wird in der Chateingabe gesucht." L.Options_Mapping_Group_Desc = "Dieses Vorkommen wird in den aktivierten Kan\195\164len ersetzt."
L.Options_Replacement_ReplaceText_Name = "Ersetzung:" L.Options_Mapping_EmptyMapping = "(keine)"
L.Options_Replacement_ReplaceText_Desc = "Jeder Suchtreffer wird mit diesem Text ersetzt." L.Options_Mapping_SearchText_Name = "Suchtext:"
L.Options_Replacement_CaseSensitive_Name = "Gro\195\159- und Kleinschreibung beachten" L.Options_Mapping_SearchText_Desc = "Dieser Text wird in der Chateingabe gesucht."
L.Options_Replacement_CaseSensitive_Desc = "Groß\195\159buchstaben werden mit Gro\195\159buchstaben ersetzt." L.Options_Mapping_ReplaceText_Name = "Ersetzung:"
L.Options_Replacement_Consolidate_Name = "Fa\195\159e aufeinanderfolgende Treffer zusammen" L.Options_Mapping_ReplaceText_Desc = "Jeder Suchtreffer wird mit diesem Text ersetzt."
L.Options_Replacement_Consolidate_Desc = "Wenn durch die Ersetzung die Zeichenfolge mehrfach hintereinander steht,|nfasse sie zu einem Vorkommen zusammen." L.Options_Mapping_ExactCase_Name = "Exakte Gro\195\159- und Kleinschreibung"
L.Options_Replacement_Delete_Name = "L\195\182schen" L.Options_Mapping_ExactCase_Desc = "Wenn gesetzt, muss die Gro\195\159- und Kleinschreibung des Suchtextes exakt \195\188berein stimmen. Anderfalls wird die Gro\195\159schreibung jedes Zeichens bei der Ersetzung \195\188bernommen."
L.Options_Replacement_Delete_Desc = "L\195\182scht diese Zuweisung." L.Options_Mapping_Consolidate_Name = "Fasse aufeinanderfolgende Treffer zusammen"
L.Options_Replacements_Delete_ConfirmText="Diese Zuweisung l\195\182schen?" L.Options_Mapping_Consolidate_Desc = "Wenn durch die Ersetzung die Zeichenfolge mehrfach hintereinander steht,|nfasse sie zu einem Vorkommen zusammen."
L.Options_Mapping_MoveUp_Name = "^"
L.Options_Mapping_MoveUp_Desc = "nach oben verschieben"
L.Options_Mapping_MoveDown_Name = "v"
L.Options_Mapping_MoveDown_Desc = "nach unten verschieben"
L.Options_Mapping_Delete_Name = "L\195\182schen"
L.Options_Mapping_Delete_Desc = "L\195\182scht diese Zuweisung."
L.Options_Mapping_Delete_ConfirmText="Diese Zuweisung l\195\182schen?"

View File

@@ -5,9 +5,14 @@ if not L then return end
L.AddonName = "Grichelde" L.AddonName = "Grichelde"
L.VersionAbbr = "v" L.VersionAbbr = "v"
L.AddonLoaded = "%s now helps you with your spelling disabilities." L.AddonLoaded = "%s now helps you with your spelling disabilities."
L.Addon_Detected_Misspelled = "The Addon 'Misspelled' has been detected and any messsage will be cleansed automatically." L.Upgrade_ToVersion = "Upgrade database to version %s."
L.Addon_Detected_WIM = "The Addon 'WIM' has been detected and any whispers will be handled from IM windows." L.Upgrade_Successful = "Upgrade successful."
-- debug
L.Debug_Options = "Options"
L.Debug_Mappings = "Mappings"
L.Debug_Mappings_Hint = "The content of this input box is used for debugging purposes only and can be copied. No input from this box will be read or processed."
L.Debug_Mappings_Found = "%d mappings found"
L.Debug_Profile = "Profile"
-- profiles -- profiles
L.Profiles_Available = "Available profiles:" L.Profiles_Available = "Available profiles:"
@@ -24,35 +29,44 @@ L.Profiles_DeleteError = "The active profile cannot be deleted!"
L.Options_Title = "%s Options" L.Options_Title = "%s Options"
L.Options_Enabled_Name = "Enabled" L.Options_Enabled_Name = "Enabled"
L.Options_Enabled_Desc = "Enables %s" L.Options_Enabled_Desc = "Enables %s"
L.Options_Minimap_Button_Name = "Show minimap button"
L.Options_Minimap_Button_Desc = "Shows or hides the button on the minimap."
L.Options_Minimap_Tooltip_Options_Left = "Left-Click"
L.Options_Minimap_Tooltip_Options_Right = "Opens or closes the options UI."
L.Options_Minimap_Tooltip_Mappings_Left = "Right-Click"
L.Options_Minimap_Tooltip_Mappings_Right = "Opens the debug window with mapping code."
L.Options_Channels_Group_Name = "Channels" L.Options_Channels_Group_Name = "Channels"
L.Options_Channels_Group_Desc = "%s is active in the following channels." L.Options_Channels_Group_Desc = "%s is active in the following channels."
L.Options_Channels_ChannelSay_Name = "Say" L.Options_Channels_Header = "Text replacement will only be done for marked channels below:"
L.Options_Channels_ChannelSay_Desc = "Activates %s in channel \"Say\"."
L.Options_Channels_ChannelEmote_Name = "Emote" L.Options_Channel_Say_Name = "Say"
L.Options_Channels_ChannelEmote_Desc = "Activates %s in channel \"Emote\"." L.Options_Channel_Say_Desc = "Activates %s in channel \"Say\"."
L.Options_Channels_ChannelYell_Name = "Yell" L.Options_Channel_Emote_Name = "Emote"
L.Options_Channels_ChannelYell_Desc = "Activates %s in channel \"Yell\"." L.Options_Channel_Emote_Desc = "Activates %s in channel \"Emote\"."
L.Options_Channels_ChannelParty_Name = "Party" L.Options_Channel_Yell_Name = "Yell"
L.Options_Channels_ChannelParty_Desc = "Activates %s in channel \"Party\"." L.Options_Channel_Yell_Desc = "Activates %s in channel \"Yell\"."
L.Options_Channels_ChannelPartyLeader_Name = "Party Leader" L.Options_Channel_Party_Name = "Party"
L.Options_Channels_ChannelPartyLeader_Desc = "Activates %s in channel \"Party Leader\"." L.Options_Channel_Party_Desc = "Activates %s in channel \"Party\"."
L.Options_Channels_ChannelGuild_Name = "Guild" L.Options_Channel_PartyLeader_Name = "Party Leader"
L.Options_Channels_ChannelGuild_Desc = "Activates %s in channel \"Guild\"." L.Options_Channel_PartyLeader_Desc = "Activates %s in channel \"Party Leader\"."
L.Options_Channels_ChannelOfficer_Name = "Officers" L.Options_Channel_Guild_Name = "Guild"
L.Options_Channels_ChannelOfficer_Desc = "Activates %s in channel \"Officers\"." L.Options_Channel_Guild_Desc = "Activates %s in channel \"Guild\"."
L.Options_Channels_ChannelRaid_Name = "Raid" L.Options_Channel_Officer_Name = "Officers"
L.Options_Channels_ChannelRaid_Desc = "Activates %s in channel \"Raid\"." L.Options_Channel_Officer_Desc = "Activates %s in channel \"Officers\"."
L.Options_Channels_ChannelRaidLeader_Name = "Raid Leader" L.Options_Channel_Raid_Name = "Raid"
L.Options_Channels_ChannelRaidLeader_Desc = "Activates %s in channel \"Raid Leader\"." L.Options_Channel_Raid_Desc = "Activates %s in channel \"Raid\"."
L.Options_Channels_ChannelRaidWarning_Name = "Raid Warning" L.Options_Channel_RaidLeader_Name = "Raid Leader"
L.Options_Channels_ChannelRaidWarning_Desc = "Activates %s in channel \"Raid Warning\"." L.Options_Channel_RaidLeader_Desc = "Activates %s in channel \"Raid Leader\"."
L.Options_Channels_ChannelInstance_Name = "Instance" L.Options_Channel_RaidWarning_Name = "Raid Warning"
L.Options_Channels_ChannelInstance_Desc = "Activates %s in channel \"Instance\"." L.Options_Channel_RaidWarning_Desc = "Activates %s in channel \"Raid Warning\"."
L.Options_Channels_ChannelBattleground_Name = "Battleground" L.Options_Channel_Instance_Name = "Instance"
L.Options_Channels_ChannelBattleground_Desc = "Activates %s in channel \"Battleground\"." L.Options_Channel_Instance_Desc = "Activates %s in channel \"Instance\"."
L.Options_Channels_ChannelWhisper_Name = "Whisper" L.Options_Channel_Battleground_Name = "Battleground"
L.Options_Channels_ChannelWhisper_Desc = "Activates %s in channel \"Whisper\"." L.Options_Channel_Battleground_Desc = "Activates %s in channel \"Battleground\"."
L.Options_Channel_Whisper_Name = "Whisper"
L.Options_Channel_Whisper_Desc = "Activates %s in channel \"Whisper\"."
L.Options_Replacements_Group_Name = "Replacements" L.Options_Replacements_Group_Name = "Replacements"
L.Options_Replacements_Group_Desc = "These lookups will be replaced in activated channels." L.Options_Replacements_Group_Desc = "These lookups will be replaced in activated channels."
@@ -61,18 +75,25 @@ L.Options_Replacements_Add_Desc = "Adds a new replacement mapping."
L.Options_Replacements_DeleteAll_Name = "Delete All" L.Options_Replacements_DeleteAll_Name = "Delete All"
L.Options_Replacements_DeleteAll_Desc = "Deletes all replacement mappings." L.Options_Replacements_DeleteAll_Desc = "Deletes all replacement mappings."
L.Options_Replacements_DeleteAll_ConfirmText = "Do you really want to delete ALL replacement mappings?" L.Options_Replacements_DeleteAll_ConfirmText = "Do you really want to delete ALL replacement mappings?"
L.Options_Replacements_Header = "All matches on the lefthand side of the arrow ( => ) will be replaced in activated channels by the text on the righthand side."
L.Options_Replacement_Group_Name = "%s => %s" .. "|nIf case sensivity is ignored, the case for each letter of the matching text is taken over when replaced."
L.Options_Replacement_Group_Desc = "This lookup will be replaced in activated channels." .. "|nConsolidation of consecutive matches prevent unaesthetic repetitions of letters introduced by replacements."
L.Options_Replacement_EmptyMapping = "(none)" .. "|nWith both default mappings active, the mapping would be \"Tossing\" => \"Ckoching\"."
L.Options_Replacement_SearchText_Name = "Search for:" L.Options_Mapping_Group_Name = "%s => %s"
L.Options_Replacement_SearchText_Desc = "This text is looked up in your chat input box." L.Options_Mapping_Group_Desc = "This lookup will be replaced in activated channels."
L.Options_Replacement_ReplaceText_Name = "Replacement:" L.Options_Mapping_EmptyMapping = "(none)"
L.Options_Replacement_ReplaceText_Desc = "Any match will be replaced with this text." L.Options_Mapping_SearchText_Name = "Search for:"
L.Options_Replacement_CaseSensitive_Name = "case sensitive" L.Options_Mapping_SearchText_Desc = "This text is looked up in your chat input box."
L.Options_Replacement_CaseSensitive_Desc = "Will not replace occurrences if cases differ." L.Options_Mapping_ReplaceText_Name = "Replacement:"
L.Options_Replacement_Consolidate_Name = "consolidate consecutive matches" L.Options_Mapping_ReplaceText_Desc = "Any match will be replaced with this text."
L.Options_Replacement_Consolidate_Desc = "If after the replacement a text sequence is repeated|ndirectly after another, treat them as one occurrence." L.Options_Mapping_ExactCase_Name = "exact case"
L.Options_Replacement_Delete_Name = "Delete" L.Options_Mapping_ExactCase_Desc = "When set, matches must be case-sensitive. Otherwise the case for each letter of the matching text is taken over when replaced."
L.Options_Replacement_Delete_Desc = "Deletes this replacement mapping." L.Options_Mapping_Consolidate_Name = "consolidate consecutive matches"
L.Options_Replacements_Delete_ConfirmText = "Delete this replacement mapping?" L.Options_Mapping_Consolidate_Desc = "If after the replacement a text sequence is repeated|ndirectly after another, treat them as one occurrence."
L.Options_Mapping_MoveUp_Name = "^"
L.Options_Mapping_MoveUp_Desc = "move up"
L.Options_Mapping_MoveDown_Name = "v"
L.Options_Mapping_MoveDown_Desc = "move down"
L.Options_Mapping_Delete_Name = "Delete"
L.Options_Mapping_Delete_Desc = "Deletes this replacement mapping."
L.Options_Mapping_Delete_ConfirmText = "Delete this replacement mapping?"

BIN
twitch/channels-de.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 108 KiB

BIN
twitch/channels-en.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 111 KiB

BIN
twitch/example-de.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 40 KiB

BIN
twitch/example-en.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 40 KiB

BIN
twitch/minimap-icon.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

BIN
twitch/profiles-de.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 128 KiB

BIN
twitch/profiles-en.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 126 KiB

BIN
twitch/replacements-de.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 111 KiB

BIN
twitch/replacements-en.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 101 KiB

BIN
twitch/send-errors.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 104 KiB

BIN
twitch/show-mappings.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 60 KiB