Version 0.9.0-rc1

- enable/disable from slash command
- matching conditions (never, always, start, end, start or end)
- support capturing groups
- import examples
- testing capabilities
- compatibility with WoW Retail
- adapted help texts
- spelling errors
master 0.9.0
Lothar Buchholz 4 years ago
parent cc4df96bac
commit bafb116bb9

@ -3,6 +3,18 @@ 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).
## Version 0.9.0-rc - 2020-07-25 [Release Candidate]
### Added
- enable/disable from slash command
- matching conditions (never, always, start, end, start or end)
- support capturing groups
- import examples
- testing capabilities
- compatibility with WoW Retail
### Changed
- adapted help texts
- spelling errors
## Version 0.8.1-beta - 2020-06-16 ## Version 0.8.1-beta - 2020-06-16
### Added ### Added
- stop replacements per mapping - stop replacements per mapping

@ -48,9 +48,9 @@ function Grichelde:OnEnable()
self:SetupSlashCommands() self:SetupSlashCommands()
-- tell the world we are listening -- tell the world we are listening
if self.db.profile.enabled then if (self.db.profile.enabled == true) then
local namePlusVersion = self:Format(self.L.AddonNamePlusVersion, self.L.AddonName, self.version) local namePlusVersion = self:Format(self.L.AddonNamePlusVersion, self.L.AddonName, self.version)
self:Print(self.L.AddonLoaded, self.COLOR_CODES.PREFIX .. namePlusVersion .. self.COLOR_CODES.CLOSE) self.F.print(self:Format(self.L.AddonLoaded, self.COLOR_CODES.PREFIX .. namePlusVersion .. self.COLOR_CODES.CLOSE))
end end
end end
@ -70,7 +70,7 @@ end
function Grichelde:HandleSlashCommand(input, ...) 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.F.nilOrEmpty(input)) then
self:ToggleOptions() self:ToggleOptions()
else else
-- handle slash ourselves -- handle slash ourselves
@ -81,6 +81,10 @@ function Grichelde:HandleSlashCommand(input, ...)
self:ToggleOptions() self:ToggleOptions()
elseif input == "profile" then elseif input == "profile" then
self:PrintProfile() self:PrintProfile()
elseif input == "on" or input == "enable" then
self:Activate()
elseif input == "off" or input == "disable" then
self:Deactivate()
else else
self:SendChatMessageOverride(input, ...) self:SendChatMessageOverride(input, ...)
end end

@ -1,9 +1,9 @@
## Interface: 11304 ## Interface: 11305
## Title: Grichelde ## Title: Grichelde
## Notes: Replaces characters from 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.8.1-beta ## Version: 0.9.0-rc
## Author: Teilzeit-Jedi ## Author: Teilzeit-Jedi
## eMail: tj@teilzeit-jedi.de ## eMail: tj@teilzeit-jedi.de
@ -30,3 +30,5 @@ GricheldeUpgrade.lua
GricheldeOptions.lua GricheldeOptions.lua
GricheldeMinimap.lua GricheldeMinimap.lua
GricheldeChat.lua GricheldeChat.lua
GricheldeTest.lua

File diff suppressed because it is too large Load Diff

@ -1,45 +1,46 @@
-- read namespace from global env -- read namespace from global env
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde or {}
-- constants and upvalues -- constants and upvalues
Grichelde.LOG_LEVEL = {} Grichelde.LOG_LEVEL = { DEBUG = 1, TRACE = 2 }
Grichelde.LOG_LEVEL.DEBUG = 1
Grichelde.LOG_LEVEL.TRACE = 2
Grichelde.MAPPING_OFFSET = 10 Grichelde.MAPPING_OFFSET = 10
Grichelde.MINIMAP_ENABLED = 1.0 Grichelde.MINIMAP_ENABLED = 1.0
Grichelde.MINIMAP_DARKENDED = 0.5 Grichelde.MINIMAP_DARKENDED = 0.5
Grichelde.ICONS = {} Grichelde.ICONS = {
Grichelde.ICONS.MOVE_UP = "Interface\\MainMenuBar\\UI-MainMenu-ScrollUpButton-Up" MOVE_UP = "Interface\\MainMenuBar\\UI-MainMenu-ScrollUpButton-Up",
Grichelde.ICONS.MOVE_UP_DISABLED = "Interface\\MainMenuBar\\UI-MainMenu-ScrollUpButton-Disabled" MOVE_UP_DISABLED = "Interface\\MainMenuBar\\UI-MainMenu-ScrollUpButton-Disabled",
Grichelde.ICONS.MOVE_DOWN = "Interface\\MainMenuBar\\UI-MainMenu-ScrollDownButton-Up" MOVE_DOWN = "Interface\\MainMenuBar\\UI-MainMenu-ScrollDownButton-Up",
Grichelde.ICONS.MOVE_DOWN_DISABLED = "Interface\\MainMenuBar\\UI-MainMenu-ScrollDownButton-Disabled" MOVE_DOWN_DISABLED = "Interface\\MainMenuBar\\UI-MainMenu-ScrollDownButton-Disabled",
Grichelde.ICONS.DELETE = "Interface\\Buttons\\UI-Panel-MinimizeButton-Up" DELETE = "Interface\\Buttons\\UI-Panel-MinimizeButton-Up",
Grichelde.ICONS.DELETE_DISABLED = "Interface\\Buttons\\UI-Panel-MinimizeButton-Disabled" DELETE_DISABLED = "Interface\\Buttons\\UI-Panel-MinimizeButton-Disabled",
}
-- colors: -- colors:
Grichelde.COLORS = {} Grichelde.COLORS = {
Grichelde.COLORS.NORMAL = _G.NORMAL_FONT_COLOR NORMAL = _G.NORMAL_FONT_COLOR,
Grichelde.COLORS.HIGHLIGHT = _G.HIGHLIGHT_FONT_COLOR HIGHLIGHT = _G.HIGHLIGHT_FONT_COLOR,
Grichelde.COLORS.RED = _G.RED_FONT_COLOR RED = _G.RED_FONT_COLOR,
Grichelde.COLORS.GREEN = _G.GREEN_FONT_COLOR GREEN = _G.GREEN_FONT_COLOR,
}
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" NORMAL = _G.NORMAL_FONT_COLOR_CODE or "|cffffd200",
Grichelde.COLOR_CODES.HIGHLIGHT = _G.HIGHLIGHT_FONT_COLOR_CODE or "|cffffffff" HIGHLIGHT = _G.HIGHLIGHT_FONT_COLOR_CODE or "|cffffffff",
Grichelde.COLOR_CODES.RED = _G.RED_FONT_COLOR_CODE or "|cffff2020" RED = _G.RED_FONT_COLOR_CODE or "|cffff2020",
Grichelde.COLOR_CODES.GREEN = _G.GREEN_FONT_COLOR_CODE or "|cff20ff20" GREEN = _G.GREEN_FONT_COLOR_CODE or "|cff20ff20",
Grichelde.COLOR_CODES.LIGHTGRAY = "|cffC0C0C0" LIGHTGRAY = "|cffC0C0C0",
Grichelde.COLOR_CODES.GRAY = _G.GRAY_FONT_COLOR_CODE or "|cff808080" GRAY = _G.GRAY_FONT_COLOR_CODE or "|cff808080",
Grichelde.COLOR_CODES.DARKGRAY = "|cff404040" DARKGRAY = "|cff404040",
Grichelde.COLOR_CODES.YELLOW = _G.YELLOW_FONT_COLOR_CODE or "|cffffff00" YELLOW = _G.YELLOW_FONT_COLOR_CODE or "|cffffff00",
Grichelde.COLOR_CODES.LIGHTYELLOW = _G.LIGHTYELLOW_FONT_COLOR_CODE or "|cffffff9a" LIGHTYELLOW = _G.LIGHTYELLOW_FONT_COLOR_CODE or "|cffffff9a",
Grichelde.COLOR_CODES.ORANGE = _G.ORANGE_FONT_COLOR_CODE or "|cffff7f3f" ORANGE = _G.ORANGE_FONT_COLOR_CODE or "|cffff7f3f",
Grichelde.COLOR_CODES.CLOSE = _G.FONT_COLOR_CODE_CLOSE or "|r" CLOSE = _G.FONT_COLOR_CODE_CLOSE or "|r",
}
Grichelde.SLASH_COMMANDS = { "gri", "grichelde" } Grichelde.SLASH_COMMANDS = { "gri", "grichelde" }
@ -113,26 +114,20 @@ Grichelde.BLIZZ_TYPE_TO_OPTIONS = {
} }
-- do not replace these patterns -- do not replace these patterns
Grichelde.IGNORE_PATTERNS_CASE_SENSITIVE = { Grichelde.IGNORE_PATTERNS = {
LINKS = {
"|[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 "|K.-|k", -- Battle.net
"|n", -- newline "|n", -- newline
},
"%(%(.-%)%)", -- (( ... )) EMOTES = {
"%(%s*ooc[%:%s].-%)", -- ( ooc )
}
-- for separate emote detection
Grichelde.EMOTE_PATTERNS = {
"%*.-%*", -- emotes * "%*.-%*", -- emotes *
"%*%*.-%*%*", -- emotes ** "%*%*.-%*%*", -- emotes **
"%<.-%>" -- emotes < > "%<.-%>", -- emotes < >
} },
SUBSTITUTES = {
Grichelde.IGNORE_PATTERNS_CASE_INSENSITIVE = {
"{rt[1-8]}", -- rumbered raid target icons, localized raid targets are handled differently
"%%n", -- player's name "%%n", -- player's name
"%%z", -- player's currnt zone "%%z", -- player's currnt zone
"%%sz", -- player's current sub-zone "%%sz", -- player's current sub-zone
@ -141,11 +136,30 @@ Grichelde.IGNORE_PATTERNS_CASE_INSENSITIVE = {
"%%f", -- name of focus target "%%f", -- name of focus target
"%%m", -- name of mouseover unit "%%m", -- name of mouseover unit
"%%p", -- name of player pet "%%p", -- name of player pet
"%%tt" -- name of player's target's target "%%tt", -- name of player's target's target
},
RAID_TARGETS = {
"{rt[1-8]}", -- rumbered raid target icons, localized raid targets are handled differently
},
LOCALIZED_RAID_TARGETS = {
"Star",
"Circle",
"Diamond",
"Triangle",
"Moon",
"Square",
"Cross",
"Skull",
},
OOC_BRACKETS = {
"%(%(.-%)%)", -- (( ... ))
"%(%s*ooc[%:%s].-%)", -- ( ooc )
},
OOC_NO_BRACKETS = {
"ooc[%:%s]",
},
} }
Grichelde.LOCALIZED_RAID_TARGETS = { "Star", "Circle", "Diamond", "Triangle", "Moon", "Square", "Cross", "Skull" }
local function nilOrEmpty(s) local function nilOrEmpty(s)
return s == nil or s:trim() == "" return s == nil or s:trim() == ""
end end
@ -154,22 +168,22 @@ local function spairs(t, orderFunc)
-- collect the keys -- collect the keys
local sortedKeys = {} local sortedKeys = {}
-- for every non-nil value -- for every non-nil value
for key, _ in Grichelde.functions.pairs(t) do for key, _ in Grichelde.F.pairs(t) do
Grichelde.functions.tInsert(sortedKeys, key) Grichelde.F.tInsert(sortedKeys, key)
end end
if orderFunc then if (orderFunc ~= nil) then
Grichelde.functions.tSort(sortedKeys, function(a, b) return orderFunc(sortedKeys, a, b) end) Grichelde.F.tSort(sortedKeys, function(a, b) return orderFunc(sortedKeys, a, b) end)
else else
-- lexicographical order -- lexicographical order
Grichelde.functions.tSort(sortedKeys) Grichelde.F.tSort(sortedKeys)
end end
-- return the iterator function -- return the iterator function
local it = 0 local it = 0
return function() return function()
it = it + 1 it = it + 1
if sortedKeys[it] then if (sortedKeys[it] ~= nil) then
return sortedKeys[it], t[sortedKeys[it]] return sortedKeys[it], t[sortedKeys[it]]
else else
return nil return nil
@ -179,23 +193,23 @@ end
local function tFilter(t, condition, extract) local function tFilter(t, condition, extract)
local filtered = {} local filtered = {}
for key, value in Grichelde.functions.pairs(t) do for key, value in Grichelde.F.pairs(t) do
local cond = false local cond = false
if condition then if (condition) then
local t = Grichelde.functions.type(condition) local t = Grichelde.F.type(condition)
if t == "function" then if (t == "function") then
cond = condition(t, key, value) cond = condition(t, key, value)
elseif t == "string" or t == "number" then elseif (t == "string") or (t == "number") then
cond = (value == condition) cond = (value == condition)
end end
end end
if cond then if (cond) then
local val = value local val = value
if extract and Grichelde.functions.type(extract) == "function" then if (extract and (Grichelde.F.type(extract) == "function")) then
val = extract(t, key, value) val = extract(t, key, value)
end end
Grichelde.functions.tInsert(filtered, val) Grichelde.F.tInsert(filtered, val)
end end
end end
return filtered return filtered
@ -203,9 +217,9 @@ end
local function tSize(t) local function tSize(t)
local size = 0 local size = 0
if (t) then if (t ~= nil) then
-- for every non-nil value -- for every non-nil value
for _, _ in Grichelde.functions.pairs(t) do for _, _ in Grichelde.F.pairs(t) do
size = size + 1 size = size + 1
end end
end end
@ -214,45 +228,117 @@ end
local function tIsEmpty(t) local function tIsEmpty(t)
if (not t) then return true end if (not t) then return true end
return Grichelde.functions.tNext(t) == nil return Grichelde.F.tNext(t) == nil
end end
local function tClone(orig) local function tClone(orig)
local orig_type = Grichelde.functions.type(orig) local orig_type = Grichelde.F.type(orig)
local copy local copy
if orig_type == 'table' then if (orig_type == 'table') then
copy = {} copy = {}
-- for every non-nil value -- for every non-nil value
for orig_key, orig_value in Grichelde.functions.tNext, orig, nil do for orig_key, orig_value in Grichelde.F.tNext, orig, nil do
copy[tClone(orig_key)] = tClone(orig_value) copy[tClone(orig_key)] = tClone(orig_value)
end end
Grichelde.functions.setmetatable(copy, tClone(Grichelde.functions.getmetatable(orig))) Grichelde.F.setmetatable(copy, tClone(Grichelde.F.getmetatable(orig)))
else -- number, string, boolean, etc else -- number, string, boolean, etc
copy = orig copy = orig
end end
return copy return copy
end end
local function isChar(word) local function getNextCharUtf8(word)
return Grichelde.functions.find(word, "[%z\65-\90\97-\122\195-\197][\128-\191]?") if ((word == nil) or (Grichelde.F.type(word) ~= "string") or (Grichelde.F.length(word) < 1)) then
return nil, nil
end
local wordLength = Grichelde.F.length(word)
local c1 = Grichelde.F.toByte(word, 1)
if (c1 > 0) and (c1 <= 127) then
-- UTF8-1
return Grichelde.F.bytes2Char(c1), Grichelde.F.sub(word, 2)
elseif (c1 >= 194) and (c1 <= 223) then
-- UTF8-2
Grichelde.F.assert(wordLength >= 2, "broken UTF-8 character")
local c2 = Grichelde.F.toByte(word, 2)
return Grichelde.F.bytes2Char(c1, c2), Grichelde.F.sub(word, 3)
elseif (c1 >= 224) and (c1 <= 239) then
-- UTF8-3
Grichelde.F.assert(wordLength >= 3, "broken UTF-8 character")
local c2 = Grichelde.F.toByte(word, 2)
local c3 = Grichelde.F.toByte(word, 3)
return Grichelde.F.bytes2Char(c1, c2, c3), Grichelde.F.sub(word, 4)
elseif (c1 >= 240) and (c1 <= 244) then
-- UTF8-4
Grichelde.F.assert(wordLength >= 4, "broken UTF-8 character")
local c2 = Grichelde.F.toByte(word, 2)
local c3 = Grichelde.F.toByte(word, 3)
local c4 = Grichelde.F.toByte(word, 4)
return Grichelde.F.bytes2Char(c1, c2, c3, c4), Grichelde.F.sub(word, 5)
else
return nil, nil
end
end
local function isLetter(word)
local char = Grichelde.F.getNextCharUtf8(word)
return (char ~= nil) and (Grichelde.F.toUpper(char) ~= Grichelde.F.toLower(char))
end end
local function isNumber(digit) local function isNumber(digit)
return Grichelde.functions.find(digit, "%d+") if ((digit == nil) or (Grichelde.F.type(digit) ~= "string") or (Grichelde.F.length(digit) < 1)) then
return false
else
return Grichelde.F.find(digit, "%d+") ~= nil
end
end end
local function isUpper(word) local function isUpper(word)
return Grichelde.functions.isChar(word) and word == Grichelde.functions.toUpper(word) if ((word == nil) or (Grichelde.F.type(word) ~= "string") or (Grichelde.F.length(word) < 1)) then
return false
elseif (Grichelde.F.toUpper(word) == Grichelde.F.toLower(word)) then
return false
else
return word == Grichelde.F.toUpper(word)
end
end end
local function isLower(word) local function isLower(word)
return Grichelde.functions.isChar(word) and word == Grichelde.functions.toLower(word) if ((word == nil) or (Grichelde.F.type(word) ~= "string") or (Grichelde.F.length(word) < 1)) then
return false
elseif (Grichelde.F.toUpper(word) == Grichelde.F.toLower(word)) then
return false
else
return word == Grichelde.F.toLower(word)
end
end end
local function isCapital(word) local function isCapital(word)
return Grichelde.functions.legnth(word) > 1 if ((word == nil) or (Grichelde.F.type(word) ~= "string") or (Grichelde.F.length(word) < 1)) then
and Grichelde.functions.isUpper(Grichelde.functions.sub(word,1,1)) return false
and Grichelde.functions.isLower(Grichelde.functions.sub(word,2)) else
local first, rest = Grichelde.F.getNextCharUtf8(word)
local isCapital = Grichelde.F.isUpper(first)
if (rest ~= nil) then
return isCapital and Grichelde.F.isLower(rest)
else
return isCapital
end
end
end
local function capitalize(word)
if ((word == nil) or (Grichelde.F.type(word) ~= "string") or (Grichelde.F.length(word) < 1)) then
return ""
else
local first, rest = Grichelde.F.getNextCharUtf8(word)
local capital = Grichelde.F.toUpper(first)
if (rest ~= nil) then
return capital .. Grichelde.F.toLower(rest)
else
return capital
end
end
end end
local function color(color, text) local function color(color, text)
@ -260,74 +346,90 @@ local function color(color, text)
end end
local function cPrefix(text) local function cPrefix(text)
return Grichelde.functions.color(Grichelde.COLOR_CODES.PREFIX, text) return Grichelde.F.color(Grichelde.COLOR_CODES.PREFIX, text)
end end
local function cYellow(text) local function cYellow(text)
return Grichelde.functions.color(Grichelde.COLOR_CODES.NORMAL, text) return Grichelde.F.color(Grichelde.COLOR_CODES.NORMAL, text)
end end
local function cGray(text) local function cGray(text)
return Grichelde.functions.color(Grichelde.COLOR_CODES.GRAY, text) return Grichelde.F.color(Grichelde.COLOR_CODES.GRAY, text)
end end
local function cDarkgray(text) local function cDarkgray(text)
return Grichelde.functions.color(Grichelde.COLOR_CODES.DARKGRAY, text) return Grichelde.F.color(Grichelde.COLOR_CODES.DARKGRAY, text)
end
local function cGreen(text)
return Grichelde.F.color(Grichelde.COLOR_CODES.GREEN, text)
end end
local function cOrange(text) local function cOrange(text)
return Grichelde.functions.color(Grichelde.COLOR_CODES.ORANGE, text) return Grichelde.F.color(Grichelde.COLOR_CODES.ORANGE, text)
end
local function cRed(text)
return Grichelde.F.color(Grichelde.COLOR_CODES.RED, text)
end end
-- faster function lookups by mapping to local refs -- faster function lookups by mapping to local refs
Grichelde.functions = {} Grichelde.F = {
Grichelde.functions.IsAddOnLoaded = _G.IsAddOnLoaded IsAddOnLoaded = _G.IsAddOnLoaded,
Grichelde.functions.assert = _G.assert assert = _G.assert,
Grichelde.functions.type = _G.type type = _G.type,
Grichelde.functions.print = _G.print print = _G.print,
Grichelde.functions.nilOrEmpty = nilOrEmpty nilOrEmpty = nilOrEmpty,
Grichelde.functions.pairs = _G.pairs pairs = _G.pairs,
Grichelde.functions.ipairs = _G.ipairs ipairs = _G.ipairs,
Grichelde.functions.spairs = spairs spairs = spairs,
Grichelde.functions.tContains = _G.tContains tContains = _G.tContains,
Grichelde.functions.tFilter = tFilter tFilter = tFilter,
Grichelde.functions.tInsert = _G.table.insert tInsert = _G.table.insert,
Grichelde.functions.tConcat = _G.table.concat tConcat = _G.table.concat,
Grichelde.functions.tSize = tSize tSize = tSize,
Grichelde.functions.tIsEmpty = tIsEmpty tIsEmpty = tIsEmpty,
Grichelde.functions.tSort = _G.table.sort tSort = _G.table.sort,
Grichelde.functions.tClone = tClone tClone = tClone,
Grichelde.functions.tNext = _G.next tNext = _G.next,
Grichelde.functions.tWipe = _G.wipe tWipe = _G.wipe,
Grichelde.functions.setmetatable = _G.setmetatable setmetatable = _G.setmetatable,
Grichelde.functions.getmetatable = _G.getmetatable getmetatable = _G.getmetatable,
Grichelde.functions.select = _G.select select = _G.select,
Grichelde.functions.unpack = _G.unpack unpack = _G.unpack,
Grichelde.functions.find = _G.string.find find = _G.string.find,
Grichelde.functions.sub = _G.string.sub sub = _G.string.sub,
Grichelde.functions.gsub = _G.string.gsub gsub = _G.string.gsub,
Grichelde.functions.match = _G.strmatch match = _G.strmatch,
Grichelde.functions.join = _G.strjoin gmatch = _G.string.gmatch,
Grichelde.functions.split = _G.strsplit join = _G.strjoin,
Grichelde.functions.toUpper = _G.strupper split = _G.strsplit,
Grichelde.functions.toLower = _G.strlower toUpper = _G.strupper,
Grichelde.functions.isChar = isChar toLower = _G.strlower,
Grichelde.functions.isNumber = isNumber getNextCharUtf8 = getNextCharUtf8,
Grichelde.functions.isUpper = isUpper isLetter = isLetter,
Grichelde.functions.isLower = isLower isNumber = isNumber,
Grichelde.functions.isCapital = isCapital isUpper = isUpper,
Grichelde.functions.color = color isLower = isLower,
Grichelde.functions.cPrefix = cPrefix isCapital = isCapital,
Grichelde.functions.cYellow = cYellow capitalize = capitalize,
Grichelde.functions.cGray = cGray color = color,
Grichelde.functions.cDarkgray = cDarkgray cPrefix = cPrefix,
Grichelde.functions.cOrange = cOrange cYellow = cYellow,
Grichelde.functions.format = _G.string.format cGray = cGray,
Grichelde.functions.rep = _G.string.rep cDarkgray = cDarkgray,
Grichelde.functions.trim = _G.strtrim cGreen = cGreen,
Grichelde.functions.length = _G.string.len cOrange = cOrange,
Grichelde.functions.lenUtf8 = _G.strlenutf8 cRed = cRed,
Grichelde.functions.toString = _G.tostring toByte = _G.string.byte,
Grichelde.functions.toNumber = _G.tonumber bytes2Char = _G.string.char,
Grichelde.functions.max = _G.math.max format = _G.string.format,
Grichelde.functions.min = _G.math.min rep = _G.string.rep,
trim = _G.strtrim,
length = _G.string.len,
lengthUtf8 = _G.strlenutf8,
toString = _G.tostring,
toNumber = _G.tonumber,
max = _G.math.max,
min = _G.math.min,
}

@ -1,11 +1,11 @@
-- read namespace from global env -- read namespace from global env
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde or {}
local pairs, tInsert, tClone, unpack, join, toString local pairs, tInsert, tClone, unpack, join, toString
= Grichelde.functions.pairs, Grichelde.functions.tInsert, Grichelde.functions.tClone, Grichelde.functions.unpack, Grichelde.functions.join, Grichelde.functions.toString = Grichelde.F.pairs, Grichelde.F.tInsert, Grichelde.F.tClone, Grichelde.F.unpack, Grichelde.F.join, Grichelde.F.toString
function Grichelde:GetDefaultConfig() function Grichelde.getDefaultConfig()
return { return {
global = {}, global = {},
profile = { profile = {
@ -24,12 +24,12 @@ function Grichelde:GetDefaultConfig()
}, },
replacements = { replacements = {
["**"] = { ["**"] = {
active = true,
order = 9999, order = 9999,
searchText = "", searchText = "",
replaceText = "", replaceText = "",
exactCase = false, exactCase = false,
consolidate = true, consolidate = true,
matchWhen = 2,
stopOnMatch = false, stopOnMatch = false,
}, },
replacement_10 = { replacement_10 = {
@ -42,13 +42,18 @@ function Grichelde:GetDefaultConfig()
searchText = "t", searchText = "t",
replaceText = "ck", replaceText = "ck",
}, },
replacement_12 = {
order = 12,
searchText = "p",
replaceText = "b",
},
} }
} }
} }
end end
function Grichelde:LoadDatabase() function Grichelde:LoadDatabase()
local db = LibStub("AceDB-3.0"):New(self.name .."DB", self:GetDefaultConfig(), 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")
@ -68,8 +73,8 @@ function Grichelde:SyncToDatabase(info, val)
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 if (info[path] ~= "mappings") then
option = option[info[path]] -- or nil option = option[info[path]]
end end
path = path + 1 path = path + 1
end end
@ -87,8 +92,8 @@ function Grichelde:ReadFromDatabase(info)
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 if (info[path] ~= "mappings") then
option = option[info[path]] -- or nil option = option[info[path]]
end end
path = path + 1 path = path + 1
end end
@ -120,7 +125,7 @@ function Grichelde:ReorderReplacements()
while count < size do while count < size do
local replName = orderToName[index] local replName = orderToName[index]
if replName and replacements[replName] then if (replName ~= nil) and (replacements[replName] ~= nil) then
self:TracePrint("ReorderReplacements : replName: %s, replTable", replName) self:TracePrint("ReorderReplacements : replName: %s, replTable", replName)
self:TracePrint(replacements[replName]) self:TracePrint(replacements[replName])
local order = Grichelde.MAPPING_OFFSET + count local order = Grichelde.MAPPING_OFFSET + count

@ -1,19 +1,19 @@
-- read namespace from global env -- read namespace from global env
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde or {}
--- add Minimap button --- add Minimap button
function Grichelde:MinimapButton() function Grichelde:MinimapButton()
local function clickHandler(_, button) local function clickHandler(_, button)
if button == 'LeftButton' then if (button == "LeftButton") then
self:ToggleOptions() self:ToggleOptions()
elseif button == 'RightButton' then elseif (button == "RightButton") then
self:ToggleActivation() self:ToggleActivation()
end end
end end
local function updateTooltip(tooltip) local function updateTooltip(tooltip)
if not tooltip or not tooltip.AddLine then return end if (tooltip == nil) or (tooltip.AddLine == nil) then return end
local tooltipTitle = self:Format(self.L.Minimap_Tooltip_Enabled, self.L.AddonName) local tooltipTitle = self:Format(self.L.Minimap_Tooltip_Enabled, self.L.AddonName)
if not self.db.profile.enabled then if not self.db.profile.enabled then
@ -35,7 +35,7 @@ function Grichelde:MinimapButton()
end end
local darkened = Grichelde.MINIMAP_ENABLED local darkened = Grichelde.MINIMAP_ENABLED
if not self.db.profile.enabled then if (self.db.profile.enabled == false) then
darkened = Grichelde.MINIMAP_DARKENDED darkened = Grichelde.MINIMAP_DARKENDED
end end
@ -63,7 +63,7 @@ function Grichelde:ToggleMinimapButton()
self.db.profile.minimapButton.hide = not self.db.profile.minimapButton.hide self.db.profile.minimapButton.hide = not self.db.profile.minimapButton.hide
self:DebugPrint("ToggleMinimapButton : hidden: ", self.db.profile.minimapButton.hide) self:DebugPrint("ToggleMinimapButton : hidden: ", self.db.profile.minimapButton.hide)
if self.db.profile.minimapButton.hide then if (self.db.profile.minimapButton.hide == true) then
self:HideMinimapButton() self:HideMinimapButton()
else else
self:ShowMinimapButton() self:ShowMinimapButton()
@ -71,37 +71,54 @@ function Grichelde:ToggleMinimapButton()
end end
function Grichelde:ShowMinimapButton() function Grichelde:ShowMinimapButton()
if self.icon then if (self.icon ~= nil) then
self.icon:Show(self.name) self.icon:Show(self.name)
end end
end end
function Grichelde:HideMinimapButton() function Grichelde:HideMinimapButton()
if self.icon then if (self.icon ~= nil) then
self.icon:Hide(self.name) self.icon:Hide(self.name)
end end
end end
function Grichelde:ToggleActivation() function Grichelde:ToggleActivation()
self.db.profile.enabled = not self.db.profile.enabled if (self.db.profile.enabled == true) then
self:Deactivate()
else
self:Activate()
end
end
function Grichelde:Activate()
self.db.profile.enabled = true
-- refresh option UI if open at the moment -- refresh option UI if open at the moment
if (self.dialog ~= nil) and (self.dialog.OpenFrames[self.name] ~= nil) then
self.dialog:SelectGroup(self.name, "enabled") self.dialog:SelectGroup(self.name, "enabled")
local namePlusVersion = self:Format(self.L.AddonNamePlusVersion, self.L.AddonName, self.version)
local statusText = self:Format(self.L.AddonLoaded, namePlusVersion)
self.dialog.OpenFrames[self.name]:SetStatusText(statusText)
end
local formatString = self.L.AddonLoaded self.ldb.iconR = Grichelde.MINIMAP_ENABLED
local darkened = Grichelde.MINIMAP_ENABLED self.ldb.iconG = Grichelde.MINIMAP_ENABLED
if not self.db.profile.enabled then self.ldb.iconB = Grichelde.MINIMAP_ENABLED
formatString = self.L.AddonUnloaded
darkened = Grichelde.MINIMAP_DARKENDED
end end
if self.dialog ~= nil and self.dialog.OpenFrames[self.name] ~= nil then function Grichelde:Deactivate()
self.db.profile.enabled = false
-- refresh option UI if open at the moment
if (self.dialog ~= nil) and (self.dialog.OpenFrames[self.name] ~= nil) then
self.dialog:SelectGroup(self.name, "enabled")
local namePlusVersion = self:Format(self.L.AddonNamePlusVersion, self.L.AddonName, self.version) local namePlusVersion = self:Format(self.L.AddonNamePlusVersion, self.L.AddonName, self.version)
local statusText = self:Format(formatString, namePlusVersion) local statusText = self:Format(self.L.AddonUnloaded, namePlusVersion)
self.dialog.OpenFrames[self.name]:SetStatusText(statusText) self.dialog.OpenFrames[self.name]:SetStatusText(statusText)
end end
self.ldb.iconR = darkened self.ldb.iconR = Grichelde.MINIMAP_DARKENDED
self.ldb.iconG = darkened self.ldb.iconG = Grichelde.MINIMAP_DARKENDED
self.ldb.iconB = darkened self.ldb.iconB = Grichelde.MINIMAP_DARKENDED
end end

@ -1,9 +1,9 @@
-- read namespace from global env -- read namespace from global env
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde or {}
local nilOrEmpty, pairs, tContains, tWipe, find, match, color, toString, toNumber local assert, nilOrEmpty, pairs, ipairs, spairs, tContains, tInsert, tClone, tWipe, find, match, cPrefix, cGray, cGreen, cOrange, cRed, format, toString, toNumber
= Grichelde.functions.nilOrEmpty, Grichelde.functions.pairs, Grichelde.functions.tContains, Grichelde.functions.tWipe, Grichelde.functions.find, Grichelde.functions.match, Grichelde.functions.color, Grichelde.functions.toString, Grichelde.functions.toNumber = Grichelde.F.assert, Grichelde.F.nilOrEmpty, Grichelde.F.pairs, Grichelde.F.ipairs, Grichelde.F.spairs, Grichelde.F.tContains, Grichelde.F.tInsert, Grichelde.F.tClone, Grichelde.F.tWipe, Grichelde.F.find, Grichelde.F.match, Grichelde.F.cPrefix, Grichelde.F.cGray, Grichelde.F.cGreen, Grichelde.F.cOrange, Grichelde.F.cRed, Grichelde.F.format, Grichelde.F.toString, Grichelde.F.toNumber
local selectedExample = 1 local selectedExample = 1
@ -206,41 +206,52 @@ function Grichelde:CreateOptionsUI()
header = { header = {
order = 2, order = 2,
type = "description", type = "description",
name = self.L.Options_Help_Examples0_Header, name = self.L.Options_Help_Examples_Header,
fontSize = "medium", fontSize = "medium",
width = 2.5, width = 2.1
}, },
dropDown = { dropDown = {
order = 3, order = 3,
type = "select", type = "select",
name = "", name = "",
--width = 1, width = 1.2,
values = { values = function(_) return self:ExtractExampleNames() end,
self.L.Options_Help_Examples1_Select, set = function(_, val) selectedExample = val end,
self.L.Options_Help_Examples2_Select,
self.L.Options_Help_Examples3_Select,
self.L.Options_Help_Examples4_Select,
self.L.Options_Help_Examples5_Select,
self.L.Options_Help_Examples6_Select,
-- self.L.Options_Help_Examples7_Select,
},
set = function(info, val) selectedExample = val end,
get = function(_) get = function(_)
self.options.args.help.args.examples.args.header.name = self.L["Options_Help_Examples" .. selectedExample .. "_Header"] self.options.args.help.args.examples.args.header.name = self.L.Options_Help_Examples[selectedExample].desc
self.options.args.help.args.examples.args.example.name = self.L["Options_Help_Examples" .. selectedExample .. "_Text"] self.options.args.help.args.examples.args.example.name = self:ExtractExampleCodes(selectedExample),
self.dialog:SelectGroup(self.name, "help", "examples", "header.name") self.dialog:SelectGroup(self.name, "help", "examples", "header.name")
return selectedExample return selectedExample
end, end,
}, },
spacer3 = { spacer3 = {
order = 3,
type = "description",
name = "",
desc = "",
width = 2.3,
},
import = {
order = 4, order = 4,
type = "execute",
width = 1,
confirm = selectedExample > 0,
name = self.L.Options_Help_Examples_Import_Name,
-- desc = self.L.Options_Help_Examples_Import_Desc,
desc = function() return format(self.L.Options_Help_Examples_Import_Desc, cPrefix(self.L.Options_Help_Examples[selectedExample].name)) end,
func = function(info) self:ImportExample(selectedExample) end,
},
spacer4 = {
order = 6,
type = "header", type = "header",
name = "", name = "",
}, },
example = { example = {
order = 5, order = 7,
type = "description", type = "description",
name = self.L.Options_Help_Examples0_Text, name = self.L.Options_Help_Examples_Text,
fontSize = "medium", fontSize = "medium",
}, },
}, },
@ -283,14 +294,14 @@ function Grichelde:CreateMapping(offset)
width = 0.15, width = 0.15,
func = function(info) self:MoveDown(info) end, func = function(info) self:MoveDown(info) end,
}, },
--[[ spacer5 = {
spacer4 = {
order = 2, order = 2,
type = "description", type = "description",
name = "", name = "",
width = 0.1, desc = "",
width = 0.05,
}, },
]] --[[
active = { active = {
order = 3, order = 3,
type = "toggle", type = "toggle",
@ -298,8 +309,40 @@ function Grichelde:CreateMapping(offset)
desc = self.L.Options_Mapping_Enabled_Desc, desc = self.L.Options_Mapping_Enabled_Desc,
width = 2.2, width = 2.2,
}, },
delete = { ]]
matchWhenLabel = {
order = 3,
type = "description",
name = self.L.Options_Mapping_MatchWhen_Name,
desc = self.L.Options_Mapping_MatchWhen_Desc,
fontSize = "medium",
width = 0.3,
},
matchWhen = {
order = 4, order = 4,
type = "select",
name = " ",
desc = self.L.Options_Mapping_MatchWhen_Desc,
values = {
self.L.Options_Mapping_MatchWhen_Select1,
self.L.Options_Mapping_MatchWhen_Select2,
self.L.Options_Mapping_MatchWhen_Select3,
self.L.Options_Mapping_MatchWhen_Select4,
self.L.Options_Mapping_MatchWhen_Select5,
self.L.Options_Mapping_MatchWhen_Select6,
},
width = 1.4,
},
spacer6 = {
order = 5,
type = "description",
name = "",
desc = "",
width = 0.1,
},
delete = {
order = 6,
type = "execute", type = "execute",
confirm = true, confirm = true,
confirmText = self.L.Options_Mapping_Delete_ConfirmText, confirmText = self.L.Options_Mapping_Delete_ConfirmText,
@ -324,21 +367,21 @@ function Grichelde:CreateMapping(offset)
}, },
exactCase = { exactCase = {
order = 20, order = 21,
type = "toggle", type = "toggle",
name = self.L.Options_Mapping_ExactCase_Name, name = self.L.Options_Mapping_ExactCase_Name,
desc = self.L.Options_Mapping_ExactCase_Desc, desc = self.L.Options_Mapping_ExactCase_Desc,
width = "full", width = "full",
}, },
consolidate = { consolidate = {
order = 21, order = 22,
type = "toggle", type = "toggle",
name = self.L.Options_Mapping_Consolidate_Name, name = self.L.Options_Mapping_Consolidate_Name,
desc = self.L.Options_Mapping_Consolidate_Desc, desc = self.L.Options_Mapping_Consolidate_Desc,
width = "full", width = "full",
}, },
stopOnMatch = { stopOnMatch = {
order = 22, order = 23,
type = "toggle", type = "toggle",
name = self.L.Options_Mapping_StopOnMatch_Name, name = self.L.Options_Mapping_StopOnMatch_Name,
desc = self.L.Options_Mapping_StopOnMatch_Desc, desc = self.L.Options_Mapping_StopOnMatch_Desc,
@ -350,7 +393,7 @@ end
function Grichelde:SetupOptions() function Grichelde:SetupOptions()
-- add DB-backed profiles to UI options -- add DB-backed profiles to UI options
local options = self:CreateOptionsUI() local options = self:CreateOptionsUI(self)
options.args.profiles = LibStub("AceDBOptions-3.0"):GetOptionsTable(self.db) options.args.profiles = LibStub("AceDBOptions-3.0"):GetOptionsTable(self.db)
options.args.profiles.order = 8 options.args.profiles.order = 8
options.args.profiles.disabled = false options.args.profiles.disabled = false
@ -363,19 +406,18 @@ function Grichelde:SetupOptions()
return options, dialog return options, dialog
end end
function Grichelde:RefreshOptions(event) function Grichelde:RefreshOptions(event, _, profileName)
self:DebugPrint("RefreshOptions : event:", event) self:DebugPrint("RefreshOptions : event:", event)
local currentProfile = color(Grichelde.COLOR_CODES.GREEN, self.db:GetCurrentProfile()) if (event == "OnNewProfile") then
if event == "OnNewProfile" then self:PrefixedPrint(self.L.Profiles_Created, cGreen(self.db:GetCurrentProfile()))
self:PrefixedPrint(self.L.Profiles_Created, currentProfile) elseif (event == "OnProfileChanged") then
elseif event == "OnProfileChanged" then self:PrefixedPrint(self.L.Profiles_Loaded, cGreen(self.db:GetCurrentProfile()))
self:PrefixedPrint(self.L.Profiles_Loaded, currentProfile) elseif (event == "OnProfileDeleted") then
elseif event == "OnProfileDeleted" then self:PrefixedPrint(self.L.Profiles_Deleted, cRed(profileName))
self:PrefixedPrint(self.L.Profiles_Deleted, currentProfile) elseif (event == "OnProfileCopied") then
elseif event == "OnProfileCopied" then self:PrefixedPrint(self.L.Profiles_Copied, cOrange(profileName))
self:PrefixedPrint(self.L.Profiles_Copied, currentProfile) elseif (event == "OnProfileReset") then
elseif event == "OnProfileReset" then self:PrefixedPrint(self.L.Profiles_Reset, cOrange(self.db:GetCurrentProfile()))
self:PrefixedPrint(self.L.Profiles_Reset, currentProfile)
else else
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
@ -386,7 +428,7 @@ end
function Grichelde:ToggleOptions() function Grichelde:ToggleOptions()
self:DebugPrint("ToggleOptions : options open: ", not not self.dialog.OpenFrames[self.name]) self:DebugPrint("ToggleOptions : options open: ", not not self.dialog.OpenFrames[self.name])
if self.dialog ~= nil and self.dialog.OpenFrames[self.name] ~= nil then if (self.dialog ~= nil) and (self.dialog.OpenFrames[self.name] ~= nil) then
self:CloseOptions() self:CloseOptions()
else else
self:OpenOptions() self:OpenOptions()
@ -394,11 +436,11 @@ function Grichelde:ToggleOptions()
end end
function Grichelde:OpenOptions() function Grichelde:OpenOptions()
if self.dialog ~= nil then if (self.dialog ~= nil) then
self.dialog:Open(self.name) self.dialog:Open(self.name)
local formatString = self.L.AddonLoaded local formatString = self.L.AddonLoaded
if not self.db.profile.enabled then if (self.db.profile.enabled == false) then
formatString = self.L.AddonUnloaded formatString = self.L.AddonUnloaded
end end
@ -409,7 +451,7 @@ function Grichelde:OpenOptions()
end end
function Grichelde:CloseOptions() function Grichelde:CloseOptions()
if self.dialog ~= nil then if (self.dialog ~= nil) then
self.dialog:Close(self.name) self.dialog:Close(self.name)
end end
end end
@ -417,7 +459,7 @@ end
--- If all replacements were disabled --- If all replacements were disabled
-- @return (boolean) -- @return (boolean)
function Grichelde:IsDisabled(info) function Grichelde:IsDisabled(info)
if info and info.option.type == "group" then if (info and (info.option.type == "group")) then
return false return false
end end
return not self.db.profile.enabled return not self.db.profile.enabled
@ -428,13 +470,13 @@ end
function Grichelde:IsMappingActive(info) function Grichelde:IsMappingActive(info)
self:TracePrint("IsMappingActive : info") self:TracePrint("IsMappingActive : info")
for i = 0, #info do for i = 0, #info do
self:TracePrint("%d = %s", i, info[i]) self:TracePrint("IsMappingActive : info[%d] = %s", i, info[i])
end end
if info and info.option.type == "group" then if (info and (info.option.type == "group")) then
return true return true
end end
if not self.db.profile.enabled then if (self.db.profile.enabled == false) then
return false return false
end end
@ -445,10 +487,10 @@ function Grichelde:IsMappingActive(info)
self:DebugPrint("IsMappingActive : \"%s\"", currentName) self:DebugPrint("IsMappingActive : \"%s\"", currentName)
self:DebugPrint(replacements[currentName]) self:DebugPrint(replacements[currentName])
if (tContains({"moveUp", "moveDown", "active", "delete"}, uiElem)) then if (tContains({"moveUp", "moveDown", "matchWhen", "delete"}, uiElem)) then
return true return true
else else
return not not replacements[currentName].active return replacements[currentName].matchWhen > 1
end end
end end
@ -457,20 +499,82 @@ function Grichelde:MappingName(info)
-- self:TracePrint(info) -- self:TracePrint(info)
local option = self.db.profile.replacements[info[2]] local option = self.db.profile.replacements[info[2]]
if nilOrEmpty(option.searchText) and nilOrEmpty(option.replaceText) then if (nilOrEmpty(option.searchText) and nilOrEmpty(option.replaceText)) then
return color(Grichelde.COLOR_CODES.GRAY, self.L.Options_Mapping_EmptyMapping) return cGray(self.L.Options_Mapping_EmptyMapping)
else else
local name = self:Format(self.L.Options_Mapping_Group_Name, option.searchText or "", option.replaceText or "") local name = self:Format(self.L.Options_Mapping_Group_Name, option.searchText or "", option.replaceText or "")
if option.active == true then if (option.matchWhen > 1) then
return name return name
else else
return color(Grichelde.COLOR_CODES.GRAY, name) return cGray(name)
end end
end end
end end
function Grichelde:ExtractExampleNames()
local exampleNames = {}
for _, example in ipairs(self.L.Options_Help_Examples) do
tInsert(exampleNames, example.name)
end
return exampleNames
end
function Grichelde:ExtractExampleCodes(num)
self:TracePrint("ExtractExampleCodes : number is: %d", num)
if (self.L.Options_Help_Examples[num] == nil) or (#self.L.Options_Help_Examples < num) then
self:DebugPrint("ExtractExampleCodes : invalid number: %d", num)
return self.L.Options_Help_Examples_Text
end
local exampleCodes = ""
for replName, replTable in spairs(self.L.Options_Help_Examples[num].replacements) do
self:TracePrint("ExtractExampleCodes : replacement: %s", replName)
self:TracePrint(replTable)
if (replTable ~= nil) and (replTable.searchText ~= nil) then
if (exampleCodes ~= "") then
exampleCodes = exampleCodes .. "|n|n"
end
exampleCodes = exampleCodes .. cPrefix(format("%s => %s", replTable.searchText, replTable.replaceText))
end
end
return exampleCodes
end
--- Create UI options for rhe given replacement table (from DB). function Grichelde:ImportExample(num)
self:TracePrint("ImportExample : number is: %d", num)
if (self.L.Options_Help_Examples[num] == nil) or (#self.L.Options_Help_Examples < num) then
self:DebugPrint("ImportExample : invalid number: %d", num)
end
local profileName = self.L.Options_Help_Examples[num].name
self:DebugPrint("ImportExample : profile name: %s", profileName)
local allProfiles = self.db:GetProfiles()
if (not tContains(allProfiles, profileName)) then
-- create new profile if not exists
self.db:SetProfile(profileName)
assert(self.db:GetCurrentProfile() == profileName, "profile was not loaded")
local exampleProfile = self.db.profile
tWipe(exampleProfile.replacements)
for replName, replTable in spairs(self.L.Options_Help_Examples[num].replacements) do
self:TracePrint("ImportExample : replacement: %s", replName)
self:TracePrint(replTable)
if (replName ~= nil) and (replTable ~= nil) and (replTable.searchText ~= nil) then
exampleProfile.replacements[replName] = tClone(replTable)
end
end
self:RefreshReplacements(self.db.profile.replacements)
else
self:ErrorPrint(self.L.Profiles_AlreadyExistsError, profileName)
end
end
--- Create UI options for the 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)
@ -480,7 +584,7 @@ function Grichelde:RefreshReplacements(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 {}
for k, _ in pairs(replacements) do for k, _ in pairs(replacements) do
if k and find(k, "^replacement_") then if (k and (find(k, "^replacement_") ~= nil)) then
replacements[k] = nil replacements[k] = nil
end end
end end
@ -505,7 +609,7 @@ function Grichelde:AddEmptyMapping()
local maxRepl = Grichelde.MAPPING_OFFSET local maxRepl = Grichelde.MAPPING_OFFSET
for replName, _ in pairs(replacements) do for replName, _ in pairs(replacements) do
local num = match(replName, "^replacement_(%d+)") local num = match(replName, "^replacement_(%d+)")
if num and maxRepl < toNumber(num) then if (num ~= nil) and (maxRepl < toNumber(num)) then
maxRepl = toNumber(num) maxRepl = toNumber(num)
end end
end end
@ -538,7 +642,7 @@ function Grichelde:MoveUp(info)
local currentOrder = toNumber(replNumber) local currentOrder = toNumber(replNumber)
-- if not on top -- if not on top
if currentOrder ~= Grichelde.MAPPING_OFFSET then if (currentOrder ~= Grichelde.MAPPING_OFFSET) then
local swapName = "replacement_" .. toString(currentOrder - 1) local swapName = "replacement_" .. toString(currentOrder - 1)
-- swap ordering -- swap ordering
@ -581,7 +685,7 @@ function Grichelde:MoveDown(info)
-- if not last element -- if not last element
self:DebugPrint("MoveDown : maxRepl: %d", maxRepl) self:DebugPrint("MoveDown : maxRepl: %d", maxRepl)
if currentOrder < maxRepl then if (currentOrder < maxRepl) then
local swapName = "replacement_" .. toString(currentOrder + 1) local swapName = "replacement_" .. toString(currentOrder + 1)
-- swap ordering -- swap ordering
@ -611,7 +715,7 @@ function Grichelde:GetMoveUpImage(info)
local _, replNumber = self:SplitOnFirstMatch(currentName, "_") local _, replNumber = self:SplitOnFirstMatch(currentName, "_")
local currentOrder = toNumber(replNumber) local currentOrder = toNumber(replNumber)
if (self:IsMappingActive(info) and currentOrder > Grichelde.MAPPING_OFFSET ) then if (self:IsMappingActive(info) and (currentOrder > Grichelde.MAPPING_OFFSET)) then
return Grichelde.ICONS.MOVE_UP return Grichelde.ICONS.MOVE_UP
else else
return Grichelde.ICONS.MOVE_UP_DISABLED return Grichelde.ICONS.MOVE_UP_DISABLED
@ -634,12 +738,12 @@ function Grichelde:GetMoveDownImage(info)
local replacements = self.db.profile.replacements or {} local replacements = self.db.profile.replacements or {}
for replName, _ in pairs(replacements) do for replName, _ in pairs(replacements) do
local num = match(replName, "^replacement_(%d+)") local num = match(replName, "^replacement_(%d+)")
if num and maxRepl < toNumber(num) then if (num ~= nil) and (maxRepl < toNumber(num)) then
maxRepl = toNumber(num) maxRepl = toNumber(num)
end end
end end
if (self:IsMappingActive(info) and currentOrder < maxRepl) then if (self:IsMappingActive(info) and (currentOrder < maxRepl)) then
return Grichelde.ICONS.MOVE_DOWN return Grichelde.ICONS.MOVE_DOWN
else else
return Grichelde.ICONS.MOVE_DOWN_DISABLED return Grichelde.ICONS.MOVE_DOWN_DISABLED

@ -0,0 +1,720 @@
-- import addon read namespace from global env
local _G = _G
local Grichelde = _G.Grichelde or {}
local find, pairs, tSize, cRed, cGreen, format = Grichelde.F.find, Grichelde.F.pairs, Grichelde.F.tSize, Grichelde.F.cRed, Grichelde.F.cGreen, Grichelde.F.format
function Grichelde:TestMatch(text, pattern)
-- disable debug print out for testing
local oldLogLevel = Grichelde.logLevel
Grichelde.logLevel = 0
local pos1, pos2, cap1, cap2, cap3, cap4, cap5, cap6, cap7, cap8, cap9 = find(text, pattern)
self:PrefixedPrint("TestMatch : text: %s, pattern: %s, pos1: %d, pos2: %d", text, pattern, pos1, pos2)
self:PrefixedPrint("TestMatch : cap1: %s, cap2: %s, cap3: %s, cap4: %s, cap5: %s, cap6: %s, cap7: %s, cap8: %s, cap9: %s", cap1, cap2, cap3, cap4, cap5, cap6, cap7, cap8, cap9)
-- restore old loglevel
Grichelde.logLevel = oldLogLevel
end
function Grichelde:RunTests()
local function test(name, replacements, testData)
local i, ok, size = 0, 0, tSize(testData)
for input, expected in pairs(testData) do
local actual = self:ReplaceText(input, replacements, false)
i = i + 1
if (actual == expected) then
ok = ok + 1
self:PrefixedPrint("Test \"%s\" (%d/%d) %s: \"%s\" => \"%s\"", name, i, size, cGreen("passed"), input, expected)
else
self:PrefixedPrint("Test \"%s\" (%d/%d) %s: \"%s\" => \"%s\", but was \"%s\"", name, i, size, cRed("failed"), input, expected, actual)
end
end
return ok, size
end
-- disable debug print out for testing
local oldLogLevel = Grichelde.logLevel
Grichelde.logLevel = 0
local ok, all, o, a = 0, 0, 0, 0
-- basic tests
o, a = test(
"fehlender Unterkiefer",
{
replacement_10 = {
order = 10,
searchText = "s",
replaceText = "ch",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
},
{
["abc"] = "abc",
["soo"] = "choo",
["oos"] = "ooch",
["oso"] = "ocho",
["sos"] = "choch",
["ssoo"] = "choo",
["osso"] = "ocho",
["ooss"] = "ooch",
["ABC"] = "ABC",
["Soo"] = "Choo",
["ooS"] = "ooCH",
["oSo"] = "oCho",
["SOS"] = "CHOCH",
["SSoo"] = "CHoo",
["OSSO"] = "OCHO",
["ooSS"] = "ooCH",
["schmeissen"] = "chmeichen",
["Schön"] = "Chön",
}
)
ok = ok + o
all = all + a
-- case sensivity and extended replacements
o, a = test(
"Zark",
{
replacement_10 = {
order = 10,
searchText = "Zark",
replaceText = "Schami",
exactCase = false,
consolidate = false,
matchWhen = 2,
stopOnMatch = false,
},
},
{
["Zark"] = "Schami",
["ZARK"] = "SCHAMI",
["Zarkilein"] = "Schamiilein",
["ZARKILEIN"] = "SCHAMIILEIN",
["Zark!"] = "Schami!",
["ZARK!"] = "SCHAMI!",
["Zark ist tot"] = "Schami ist tot",
["ZARK ist tot"] = "SCHAMI ist tot",
["Zark ist der Tod"] = "Schami ist der Tod",
["ZARK IST DER TOD"] = "SCHAMI IST DER TOD",
}
)
ok = ok + o
all = all + a
-- start/end of sentence/words
o, a = test(
"wann",
{
replacement_10 = {
order = 10,
searchText = "bcd",
replaceText = "efg",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "uio",
replaceText = "bnm",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "hij",
replaceText = "klm",
exactCase = false,
consolidate = true,
matchWhen = 4,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "nop",
replaceText = "qrs",
exactCase = false,
consolidate = true,
matchWhen = 5,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "tuv",
replaceText = "wxy",
exactCase = false,
consolidate = true,
matchWhen = 6,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "wer",
replaceText = "sdf",
exactCase = false,
consolidate = true,
matchWhen = 7,
stopOnMatch = false,
},
},
{
-- replacement_10
["bcd"] = "efg",
["abcdz"] = "aefgz",
["abcd"] = "aefg",
["bcdz"] = "efgz",
-- replacement_10
["uio"] = "bnm",
["auioz"] = "auioz",
["auio"] = "auio",
["uioz"] = "uioz",
-- replacement_12
["hij"] = "klm",
["ahijz"] = "ahijz",
["ahij"] = "ahij",
["hijz"] = "klmz",
-- replacement_13
["nop"] = "qrs",
["anopz"] = "anopz",
["anop"] = "aqrs",
["nopz"] = "nopz",
-- replacement_14
["tuv"] = "wxy",
["atuvz"] = "atuvz",
["atuv"] = "awxy",
["tuvz"] = "wxyz",
-- replacement_15
["wer"] = "wer",
["awerz"] = "asdfz",
["awer"] = "awer",
["werz"] = "werz",
}
)
ok = ok + o
all = all + a
o, a = test(
"consolidate",
{
replacement_10 = {
order = 10,
searchText = "a",
replaceText = "b",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "b",
replaceText = "c",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "d",
replaceText = "e",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
},
{
["aaa"] = "c",
["abb"] = "c",
["abc"] = "c",
["bbc"] = "c",
["ace"] = "ce",
["abe"] = "ce",
["bbe"] = "ce",
["ece"] = "ece",
["ede"] = "ee",
}
)
ok = ok + o
all = all + a
-- stop on match
o, a = test(
"stopOnMatch",
{
replacement_10 = {
order = 10,
searchText = "a",
replaceText = "b",
exactCase = false,
consolidate = false,
matchWhen = 2,
stopOnMatch = true,
},
replacement_11 = {
order = 11,
searchText = "b",
replaceText = "c",
exactCase = false,
consolidate = false,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "c",
replaceText = "d",
exactCase = false,
consolidate = false,
matchWhen = 2,
stopOnMatch = true,
},
},
{
["aaa"] = "bbb",
["abc"] = "bbc",
["bbc"] = "ddd",
["bca"] = "bcb",
}
)
ok = ok + o
all = all + a
o, a = test(
"Jar Jar Binks (DE)",
{
replacement_10 = {
order = 10,
searchText = "ver",
replaceText = "va",
exactCase = false,
consolidate = false,
matchWhen = 4,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "en",
replaceText = "'n",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "er",
replaceText = "a",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "(%w?)ich",
replaceText = "%1ichse",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "(d?m?)ir",
replaceText = "%1ichse",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "du",
replaceText = "du da",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_16 = {
order = 16,
searchText = "er",
replaceText = "erse",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_17 = {
order = 17,
searchText = "sie",
replaceText = "sie da",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_18 = {
order = 18,
searchText = "wir",
replaceText = "wirse",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_19 = {
order = 19,
searchText = "ihr",
replaceText = "ihrse",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_20 = {
order = 20,
searchText = "nicht",
replaceText = "nich",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_21 = {
order = 21,
searchText = "die",
replaceText = "de",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
},
{
["Ich kann dich verstehen."] = "Ichse kann dichse vasteh'n.",
["Wir haben sie die ganze Zeit über nicht verstanden"] = "Wirse hab'n sie da de ganze Zeit üba nich vastand'n",
}
)
ok = ok + o
all = all + a
o, a = test(
"Stottern",
{
replacement_10 = {
order = 10,
searchText = "^([^aeiouy]*)([aeiouy])",
replaceText = "%1%2-%1%2-%1%2",
exactCase = false,
consolidate = true,
matchWhen = 4,
stopOnMatch = false,
},
},
{
["Ich mag dich."] = "I-I-Ich mag dich.",
["Dich mag ich."] = "Di-Di-Dich mag ich.",
}
)
ok = ok + o
all = all + a
o, a = test(
"trollifier",
{
replacement_10 = {
order = 10,
searchText = "(%w)(%p?)$",
replaceText = "%1, mon%2",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "th",
replaceText = "d",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "what are you",
replaceText = "whatcha",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "your?s?",
replaceText = "ya",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "going to",
replaceText = "gonna",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "want to",
replaceText = "wanna",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_16 = {
order = 16,
searchText = "ing",
replaceText = "in'",
exactCase = false,
consolidate = true,
matchWhen = 5,
stopOnMatch = false,
},
},
{
["What are you going to do when they come for you?"] = "Whatcha gonna do when dey come for ya, mon?",
["That's what young people are doing"] = "Dat's what young people are doin', mon",
}
)
ok = ok + o
all = all + a
o, a = test(
"Jar Jar Binks (EN)",
{
replacement_10 = {
order = 10,
searchText = "me",
replaceText = "mesa",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "I am",
replaceText = "Mesa",
exactCase = true,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "I'm",
replaceText = "Mesa",
exactCase = true,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "I",
replaceText = "Me",
exactCase = true,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "you are",
replaceText = "yousa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "you're",
replaceText = "yousa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_16 = {
order = 16,
searchText = "your",
replaceText = "yous",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_17 = {
order = 17,
searchText = "(s?)he is",
replaceText = "%1hesa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_18 = {
order = 18,
searchText = "(s?)he's",
replaceText = "%1hesa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_19 = {
order = 19,
searchText = "they",
replaceText = "daysa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_20 = {
order = 20,
searchText = "them",
replaceText = "them-sa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_21 = {
order = 21,
searchText = "ing",
replaceText = "in'",
exactCase = false,
consolidate = true,
matchWhen = 5,
stopOnMatch = false,
},
replacement_22 = {
order = 22,
searchText = "the",
replaceText = "da",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_23 = {
order = 23,
searchText = "th",
replaceText = "d",
exactCase = false,
consolidate = false,
matchWhen = 2,
stopOnMatch = false,
},
replacement_24 = {
order = 24,
searchText = "yes",
replaceText = "yesa",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_25 = {
order = 25,
searchText = "oka?y?",
replaceText = "okeeday",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
},
{
["I hear your voice through the thrilling grapewine."] = "Me hear yous voice drough da drillin' grapewine.",
["They gave them their OK"] = "Daysa gave dem-sa deir OKEEDAY",
}
)
ok = ok + o
all = all + a
o, a = test(
"old-fashioned",
{
replacement_10 = {
order = 10,
searchText = "oi",
replaceText = "oy",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "([^aeiou]*)([aeiou])",
replaceText = "%1%2e",
exactCase = false,
consolidate = true,
matchWhen = 5,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "yours",
replaceText = "thy",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "youe",
replaceText = "tho",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
},
{
["Do you want to kill yours?"] = "Doe thou want toe kill thy?",
}
)
ok = ok + o
all = all + a
if (ok == all) then
self:PrefixedPrint("All %d tests %s", all, cGreen("passed"))
else
self:PrefixedPrint("%d test %s, %d tests %s", all - ok, cRed("failed"), ok, cGreen("passed"))
end
-- restore old loglevel
Grichelde.logLevel = oldLogLevel
end

@ -1,8 +1,9 @@
-- read namespace from global env -- read namespace from global env
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde or {}
local pairs, find, color, cOrange, toNumber = Grichelde.functions.pairs, Grichelde.functions.find, Grichelde.functions.color, Grichelde.functions.cOrange, Grichelde.functions.toNumber local pairs, find, cGreen, cOrange, cRed, toNumber
= Grichelde.F.pairs, Grichelde.F.find, Grichelde.F.cGreen, Grichelde.F.cOrange, Grichelde.F.cRed, Grichelde.F.toNumber
function Grichelde:Upgrade_To_v060() function Grichelde:Upgrade_To_v060()
self:PrefixedPrint(self.L.Upgrade_ToVersion, cOrange("0.6.0")) self:PrefixedPrint(self.L.Upgrade_ToVersion, cOrange("0.6.0"))
@ -67,45 +68,89 @@ function Grichelde:Upgrade_To_v080()
return 0, 8, 0 return 0, 8, 0
end end
function Grichelde:Upgrade_To_v090()
self:PrefixedPrint(self.L.Upgrade_ToVersion, cOrange("0.9.0"))
local replacements = self.db.profile.replacements or {}
self:DebugPrint("Upgrade_To_v090 : old replacements")
self:DebugPrint(replacements)
for _, replTable in pairs(replacements) do
if (replTable["active"] == true) then
replTable["matchWhen"] = 2
else
replTable["matchWhen"] = 1
end
replTable["active"] = nil
end
self:DebugPrint("Upgrade_To_v090 : new replacements")
self:DebugPrint(self.db.profile)
return 0, 9, 0
end
function Grichelde:UpgradeDatabase() function Grichelde:UpgradeDatabase()
local dbVersion = self.db.global.version or "0.0.0" local dbVersion = self.db.global.version or "0.0.0"
self:DebugPrint("Database version:", dbVersion) self:DebugPrint("Database version:", dbVersion)
local _, _, maj, min, pat = find(dbVersion, "(%d+)%.(%d+)%.(%d+).*") local dbMajor, dbMinor, dbPatch = self:ParseVersion(dbVersion)
local major, minor, patch = toNumber(maj) or 0, toNumber(min) or 0, toNumber(pat) or 0 local gMajor, gMinor, gPatch = self:ParseVersion(self.version)
local downGrade = false
if (dbMajor > gMajor) then
downGrade = true
elseif dbMajor == gMajor then
if (dbMinor > gMinor) then
downGrade = true
elseif dbMinor == gMinor then
if (dbPatch > gPatch) then
downGrade = true
end
end
end
if downGrade then
self:PrefixedPrint(cRed(self.L.Downgrade_Detected), self.L.AddonName)
else
local upgrade = 0 local upgrade = 0
local error = false local error = false
if major == 0 then if (dbMajor == 0) then
if minor < 6 then if (dbMinor < 6) then
upgrade = upgrade + 1 upgrade = upgrade + 1
major, minor, patch = self:Upgrade_To_v060(dbVersion) dbMajor, dbMinor, dbPatch = self:Upgrade_To_v060(dbVersion)
end end
if minor < 7 then if (dbMinor < 7) then
upgrade = upgrade + 1 upgrade = upgrade + 1
major, minor, patch = self:Upgrade_To_v070(dbVersion) dbMajor, dbMinor, dbPatch = self:Upgrade_To_v070(dbVersion)
end end
if minor == 7 then if (dbMinor == 7) then
if patch < 2 then if (dbPatch < 2) then
upgrade = upgrade + 1 upgrade = upgrade + 1
major, minor, patch = self:Upgrade_To_v072(dbVersion) dbMajor, dbMinor, dbPatch = self:Upgrade_To_v072(dbVersion)
end end
end end
if minor < 8 then if (dbMinor < 8) then
upgrade = upgrade + 1 upgrade = upgrade + 1
major, minor, patch = self:Upgrade_To_v080(dbVersion) dbMajor, dbMinor, dbPatch = self:Upgrade_To_v080(dbVersion)
end
if (dbMinor < 9) then
upgrade = upgrade + 1
dbMajor, dbMinor, dbPatch = self:Upgrade_To_v090(dbVersion)
end end
end end
if upgrade == 0 then if (upgrade == 0) or (error == false) then
self:DebugPrint("Database up-to-date")
-- bump version number even if no update is required -- bump version number even if no update is required
self.db.global.version = self.version self.db.global.version = self:Format("%d.%d.%d", gMajor, gMinor, gPatch)
end
if (upgrade == 0) then
self:DebugPrint("Database up-to-date")
elseif (error == false) then
self:PrefixedPrint(cGreen(self.L.Upgrade_Successful))
else else
if not error then self:PrefixedPrint(cRed(self.L.Upgrade_Error))
self.db.global.version = self.version
self:PrefixedPrint(color(Grichelde.COLOR_CODES.GREEN, self.L.Upgrade_Successful))
end end
end end
end end

@ -1,18 +1,33 @@
-- import addon read namespace from global env -- import addon read namespace from global env
local _G = _G local _G = _G
local Grichelde = _G.Grichelde local Grichelde = _G.Grichelde or {}
local type, print, pairs, tSize, select, unpack, find, color, cGray, cDarkgray, cPrefix, format, rep, toString local type, print, pairs, tSize, select, unpack, find, sub, gsub, cGray, cDarkgray, cRed, cPrefix, format, rep, toString, toNumber
= Grichelde.functions.type, Grichelde.functions.print, Grichelde.functions.pairs, Grichelde.functions.tSize, Grichelde.functions.select, Grichelde.functions.unpack, Grichelde.functions.find, Grichelde.functions.color, Grichelde.functions.cGray, Grichelde.functions.cDarkgray, Grichelde.functions.cPrefix, Grichelde.functions.format, Grichelde.functions.rep, Grichelde.functions.toString = Grichelde.F.type, Grichelde.F.print, Grichelde.F.pairs, Grichelde.F.tSize, Grichelde.F.select, Grichelde.F.unpack, Grichelde.F.find, Grichelde.F.sub, Grichelde.F.gsub, Grichelde.F.cGray, Grichelde.F.cDarkgray, Grichelde.F.cRed, Grichelde.F.cPrefix, Grichelde.F.format, Grichelde.F.rep, Grichelde.F.toString, Grichelde.F.toNumber
function Grichelde:ParseVersion(version)
local _, _, major, minor, patch, ext = find(version, "(%d+)%.(%d+)%.(%d+)(.*)")
local preBuild, build = ext, ""
if (sub(ext, 1, 1) == "-") then
local b = find(ext, "+", 2)
if (b ~= nil) then
preBuild = sub(ext, 1, b)
build = sub(ext, b + 1)
else
preBuild = sub(ext, 1, b)
end
end
return toNumber(major) or 0, toNumber(minor) or 0, toNumber(patch) or 0, preBuild, build
end
-- show strings differently to distinguish them from numbers -- show strings differently to distinguish them from numbers
local function plainValue(val) function Grichelde:plainValue(val)
if val == nil then if (val == nil) then
return "<nil>" return "<nil>"
elseif type(val) == "string" then elseif (type(val) == "string") then
return '"' .. val .. '"' return '"' .. val .. '"'
elseif type(val) == "table" then elseif (type(val) == "table") then
if tSize(val) > 0 then if (tSize(val) > 0) then
return toString(val) return toString(val)
else else
return "{}" return "{}"
@ -23,33 +38,36 @@ local function plainValue(val)
end end
--- Prints any value to default channel, do NOT return a string. --- Prints any value to default channel, do NOT return a string.
local function tPrint(val, indent, known, printFunc) function Grichelde:tPrint(val, indent, known, printFunc)
local printF = printFunc or print local printF = printFunc or print
indent = indent or 0 indent = indent or 0
known = known or {} known = known or {}
if val == nil then if (val == nil) then
printF(rep(" ", indent) .. "<nil>") printF(rep(" ", indent) .. "<nil>")
elseif type(val) == "string" then elseif (type(val) == "string") then
printF(rep(" ", indent) .. "\"" .. val .. "\"") printF(rep(" ", indent) .. "\"" .. val .. "\"")
elseif type(val) == "table" then elseif (type(val) == "table") then
if tSize(val) > 0 then if (tSize(val) > 0) then
for key, value in pairs(val) do for key, value in pairs(val) do
if value == nil then if (value == nil) then
printF(rep(" ", indent) .. plainValue(key) .. " = <nil>") printF(rep(" ", indent) .. self:plainValue(key) .. " = <nil>")
elseif type(value) == "table" then elseif (type(value) == "table") then
printF(rep(" ", indent) .. plainValue(key) .. " = {") printF(rep(" ", indent) .. self:plainValue(key) .. " = {")
if tSize(value) > 0 then if (tSize(value) > 0) then
if not known[value] then if not known[value] then
tPrint(value, indent + 4, known, printF) self:tPrint(value, indent + 4, known, printF)
known[value] = true known[value] = true
else else
printF("<known table> " .. plainValue(value)) printF("<known table> " .. self:plainValue(value))
end end
end end
printF(rep(" ", indent) .. "}") printF(rep(" ", indent) .. "}")
else else
printF(rep(" ", indent) .. plainValue(key) .. " = " .. plainValue(value)) local k = self:plainValue(key)
local v = self:plainValue(value)
--print( "k: " .. k .. ", v: " ..v)
printF(rep(" ", indent) .. k .. " = " .. v)
end end
end end
else else
@ -62,7 +80,7 @@ end
--- Splits at first word of a text line --- Splits at first word of a text line
function Grichelde:SplitOnFirstMatch(text, delimPattern, start) 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:TracePrint("SplitOnFirstMatch : text: %s, pattern: %s, start: %d", text, pattern, start) self:TracePrint("SplitOnFirstMatch : text: %s, pattern: %s, start: %d", text, pattern, start)
@ -88,63 +106,67 @@ function Grichelde:TestMatch(text, pattern)
end end
function Grichelde:Format(message, ...) function Grichelde:Format(message, ...)
if ( not message ) then if (message == nil) then
return "<nil>" return "<nil>"
elseif type(message) == "string" then elseif (type(message) == "string") then
if ( not find(message, "%%")) then if (find(message, "%%") == nil) then
--print("message: ", message)
--print("...: ", ...)
return message, ... return message, ...
else else
local l = select("#", ...) local l = select("#", ...)
if l > 0 then if (l > 0) then
-- sanitize nil values in vararg -- sanitize nil values in vararg
local packed = { ... } local packed = { ... }
for i = 1, l do for i = 1, l do
packed[i] = toString(packed[i]) or "nil" packed[i] = toString(packed[i]) or "nil"
--print("packed[i] = ", packed[i])
--packed[i] = gsub(packed[i], "%%(%d)", "%%%%%1")
end end
-- print("packed = ", packed) -- print("packed = ", packed)
-- 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
-- Manually set count as unpack() stops on nil (bug with #table) -- Manually set count as unpack() stops on nil (bug with #table)
return format(message, unpack(packed, 1, l)) return format(message, unpack(packed, 1, l))
else
return message
end end
end end
end end
end end
--- deprecated
function Grichelde:Print(...)
print(self:Format(...))
end
function Grichelde:PrefixedPrint(...) function Grichelde:PrefixedPrint(...)
print(cPrefix(self.L.AddonName) .. ":", self:Format(...)) print(cPrefix(self.L.AddonName) .. ":", self:Format(...))
end end
function Grichelde:ErrorPrint(...) function Grichelde:ErrorPrint(...)
print(cPrefix(self.L.AddonName) .. ": " .. color(self.COLOR_CODES.RED, self:Format(...))) print(cPrefix(self.L.AddonName) .. ": " .. cRed(self:Format(...)))
end end
function Grichelde:DebugPrint(obj, ...) function Grichelde:DebugPrint(obj, ...)
self:LogPrint(Grichelde.LOG_LEVEL.DEBUG, function(...) if (self.logLevel >= Grichelde.LOG_LEVEL.DEBUG) then
self:LogPrint(function(...)
print(cGray(self.L.AddonName) .. ":", self:Format(...)) print(cGray(self.L.AddonName) .. ":", self:Format(...))
end, obj, ...) end, obj, ...)
end end
end
function Grichelde:TracePrint(obj, ...) function Grichelde:TracePrint(obj, ...)
self:LogPrint(Grichelde.LOG_LEVEL.TRACE, function(...) if (self.logLevel >= Grichelde.LOG_LEVEL.TRACE) then
self:LogPrint(function(...)
print(cDarkgray(self.L.AddonName) .. ":", self:Format(...)) print(cDarkgray(self.L.AddonName) .. ":", self:Format(...))
end, obj, ...) end, obj, ...)
end end
end
function Grichelde:LogPrint(logLevel, printFunc, obj, ...) function Grichelde:LogPrint(printFunc, obj, ...)
if (self.logLevel >= logLevel) then
local printF = printFunc or print local printF = printFunc or print
if obj == nil then if obj == nil then
printF("<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 (find(obj, "%%") == nil) then
printF(obj, ...) printF(obj, ...)
else else
-- sanitize nil values in vararg -- sanitize nil values in vararg
@ -159,11 +181,10 @@ function Grichelde:LogPrint(logLevel, printFunc, obj, ...)
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)
printF(fmtMsg) printF(fmtMsg)
end end
elseif type(obj) == "table" then elseif (type(obj) == "table") then
tPrint(obj, 0, {}, printF) self:tPrint(obj, 0, {}, printF)
else else
printF(plainValue(obj)) printF(self:plainValue(obj))
end
end end
end end
end end
@ -196,7 +217,7 @@ end
function Grichelde:ToogleMappings() function Grichelde:ToogleMappings()
local AceGUI = LibStub("AceGUI-3.0") local AceGUI = LibStub("AceGUI-3.0")
if self.debugFrame then if (self.debugFrame ~= nil) then
AceGUI:Release(self.debugFrame) AceGUI:Release(self.debugFrame)
self.debugFrame = nil self.debugFrame = nil
else else
@ -231,7 +252,7 @@ function Grichelde:ToogleMappings()
local configBox = AceGUI:Create("MultiLineEditBox"); local configBox = AceGUI:Create("MultiLineEditBox");
configBox:SetLabel("") configBox:SetLabel("")
local text = "" local text = ""
tPrint(replacements, 0, {}, function(s) text = text .. s .. "|n" end) self:tPrint(replacements, 0, {}, function(s) text = text .. s .. "|n" end)
configBox:SetText(text) configBox:SetText(text)
configBox:SetFullWidth(true) configBox:SetFullWidth(true)
--configBox:SetFullHeight(true) --configBox:SetFullHeight(true)

@ -1,8 +1,8 @@
# Grichelde - Text replacer # Grichelde - Text replacer
Grichelde is a WoW Classic Addon that replaces any characters or words you typed in a chatbox with any replacement text or word you specified. Grichelde is a WoW Classic Addon that replaces any characters or words you typed in a chatbox according to your replacement rules and text.
You can define any search and replace text arbitrarily. The replacement is done **before** the text is send to others/in the target channel. The replacement is done **before** the text is send to others/in the target channel. It does **not** change text others have written in the chat/channel,
It does **not** change txt others have written in the chat/channel. but the text they will see **from you** when you sent it.
Its purpose it to simulate a speaking disability of your toon and hereby strengthen immersion in roleplay. Its purpose it to simulate a speaking disability of your toon and hereby strengthen immersion in roleplay.
Initially started as a helper addon for a roleplaying friend, Grichelde can be used for much more, like Initially started as a helper addon for a roleplaying friend, Grichelde can be used for much more, like
@ -12,6 +12,9 @@ Initially started as a helper addon for a roleplaying friend, Grichelde can be u
* write out abbreviations for you * write out abbreviations for you
* create hilarious moments during roleplay * create hilarious moments during roleplay
However it does not replace slash commands with other slash commands, it only works for chat text. It you want to replace commands,
please look at more sophisticated chat addons like [Prat](https://www.curseforge.com/wow/addons/prat-3-0).
## Disclaimer ## Disclaimer
#### No Warranty #### No Warranty
The addon is provided "AS IS" and comes without warranty of any kind of function or correctness (for more details, consult the GPL 3 license). The addon is provided "AS IS" and comes without warranty of any kind of function or correctness (for more details, consult the GPL 3 license).
@ -35,19 +38,19 @@ Only slash commands, item links, textures, placeholders and ooc-markers are excl
#### My replacement is not taken. #### 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, After entering a search or replacement text, you see a button "Okay" next to your input. This is **not** a validation message,
but the save button for text. This is a restriction from the UI library and can be seen in other addons as well. but the save button for text. This is a restriction from the UI library and can be seen in other addons as well.
Please click on "Okay" button to save the input permanently. Please click on "Okay" to save the input permanently.
If it still does not work or gives you errors, please read the next question. If it still does not work or gives you errors, please read the next question.
#### I have to disable the Addon frequently. Is there a more elegant solution #### I have to disable the Addon frequently. Is there a more elegant solution?
Actually there are two solutions: Actually there are two solutions:
1. A right-click on the minimap button will disable Grichelde instantly. Right-click a second time will activate it again. Easy, isn't it? 1. A right-click on the minimap button will disable Grichelde instantly. Right-click a second time will activate it again. Easy, isn't it?
2. You can disable Grichelde permanently and forcefully replace the sentence in the chatbox. I call it "Grichelde-On-Demand" :) 2. You can disable Grichelde permanently and forcefully replace a sentence in the chatbox with manual override. I call it "Grichelde-On-Demand" :)
In the chatbox put `/gri` or `/grichelde` in front of your typed text, you can also include the target channel, In the chatbox put `/gri` or `/grichelde` in front of your typed text, you can also include the target channel,
i.e. `/gri /guild hello there` and Grichelde will apply the active replacements even if guild channel or Grichelde was disabled. i.e. `/gri /guild hello there` and Grichelde will apply the active replacements and rules even if guild channel or Grichelde was disabled.
#### I get errors, what should I do? #### I get errors, what should I do?
@ -56,13 +59,15 @@ You can bring up a small windows with your mapping by entering the `/gri mapping
#### Why that strange name? #### Why that strange name?
Grichelde is the name of an Undead rogue without a jaw, who was played in RP sessions with a guild member. Grichelde is the name of an Undead rogue without a jaw, who was played in RP sessions by a guild member.
She started replacing "s" and "t" letters manually for each line in the chat, which became cumbersome over time. She started replacing `s` and `t` letters manually for each word in the chat, which became cumbersome over time.
(If you ever wondered how an Undead without a jaw sounds like, its really hilarious, you should try it.) (If you ever wondered how an Undead without a jaw sounds like, its really hilarious, you should try it.)
Without spelling errors, "Griselde" in German is an old-fashioned female first name. Without spelling errors, "Griselde" in German is an old-fashioned female first name.
#### I'm a Pro. Does it support regular expressions? #### I'm a Pro. Does it support regular expressions?
This is actually an unofficial feature. In general the searchText is passed in as Lua, so yes regex can be used in lookups. RegEx are very powerful search and replacement patterns commonly used in programming. Technically all searches the addon performs on the input are done
There are two caveats: first, Lua does not support PCRE syntax, as it would bloat Lua's simplicity and performance (read [here](http://www.lua.org/pil/20.1.html) why). with regular expression methods, however Lua unfortunately does not support full [PCRE](https://en.wikipedia.org/wiki/Perl_Compatible_Regular_Expressions) syntax and is very limited (read [here](http://www.lua.org/pil/20.1.html) why).
Secondly, Grichelde does not support capture groups in the replacement text, so matches get lost. Nevertheless some patterns can be used like the anchors start of line `^` or end of line `$`, capturing groups `(hello) (world)`, character classes like numbers `%d` or (inversed) sets `[^%p]`.
Capture groups can be accessed in the replacement text with `%<number>` like in `%2 %1`. Also there are some mappings using RegEx in the Example section.

@ -6,9 +6,9 @@ local Grichelde = _G.Grichelde
local L = LibStub('AceLocale-3.0'):NewLocale(AddonName, 'deDE') local L = LibStub('AceLocale-3.0'):NewLocale(AddonName, 'deDE')
if not L then return end if not L then return end
local cYellow = Grichelde.functions.cYellow local cYellow = Grichelde.F.cYellow
local cGray = Grichelde.functions.cGray local cGray = Grichelde.F.cGray
local cPrefix = Grichelde.functions.cPrefix local cPrefix = Grichelde.F.cPrefix
-- system messages -- system messages
L.AddonName = "Grichelde" L.AddonName = "Grichelde"
@ -17,6 +17,8 @@ L.AddonLoaded = "%s unterst\195\188tzt Euch jetzt bei euren Sprachschwierigkeite
L.AddonUnloaded = "%s wartet geduldig Euch weiter unterst\195\188tzen zu d\195\188rfen." L.AddonUnloaded = "%s wartet geduldig Euch weiter unterst\195\188tzen zu d\195\188rfen."
L.Upgrade_ToVersion = "Hebe Databank auf Version %s an." L.Upgrade_ToVersion = "Hebe Databank auf Version %s an."
L.Upgrade_Successful = "Upgrade erfolgreich." L.Upgrade_Successful = "Upgrade erfolgreich."
L.Upgrade_Error = "Upgrade fehlgeschlagen!"
L.Downgrade_Detected = "Downgrade erkannt, %s kann sich m\195\182glichweise fehlerhaft verhalten!"
-- debug -- debug
L.Debug_Options = "Optionen" L.Debug_Options = "Optionen"
@ -41,6 +43,7 @@ L.Profiles_Copied = "Einstellungen von Profil %s \195\188bernommen."
L.Profiles_Reset = "Profil %s zur\195\188ckgesetzt." L.Profiles_Reset = "Profil %s zur\195\188ckgesetzt."
L.Profiles_Invalid = "Ung\195\188ltiges Profil %s!" L.Profiles_Invalid = "Ung\195\188ltiges Profil %s!"
L.Profiles_DeleteError = "Das aktive Profil kann nicht gel\195\182scht werden!" L.Profiles_DeleteError = "Das aktive Profil kann nicht gel\195\182scht werden!"
L.Profiles_AlreadyExistsError = "Das Profil %s existiert bereits!"
-- minimap -- minimap
L.Minimap_Tooltip_Enabled = "%s" L.Minimap_Tooltip_Enabled = "%s"
@ -100,8 +103,19 @@ L.Options_Replacements_Header = "Die Vorkommen links vom Pfeil \"=>\" werden in
L.Options_Mapping_Group_Name = "%s => %s" L.Options_Mapping_Group_Name = "%s => %s"
L.Options_Mapping_Group_Desc = "Dieses Vorkommen wird in den aktivierten Kan\195\164len ersetzt." L.Options_Mapping_Group_Desc = "Dieses Vorkommen wird in den aktivierten Kan\195\164len ersetzt."
L.Options_Mapping_EmptyMapping = "(keine)" L.Options_Mapping_EmptyMapping = "(keine)"
L.Options_Mapping_Enabled_Name = "Aktiv" L.Options_Mapping_MoveUp_Name = "^"
L.Options_Mapping_Enabled_Desc = "Diese Ersetzung wird durchgef\195\188hrt" 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_MatchWhen_Name = "wann:"
L.Options_Mapping_MatchWhen_Desc = "F\195\188hrt die Ersetzung nur durch, wenn der Suchtext |nirgendwo vorkommt (<immer>), |nwenn der Suchtext <als ganzes Wort> \195\188bereinstimmt, |n<nur am Anfang eines Worts>, |noder <nur am Ende eines Worts>, |noder <nur am Anfang oder Ende eines Worts> aber nicht dazwischen, |noder nur in der Wortmitte aber <nie am Anfang und Ende eines Worts>."
L.Options_Mapping_MatchWhen_Select1 = "nie (deaktivert)"
L.Options_Mapping_MatchWhen_Select2 = "immer"
L.Options_Mapping_MatchWhen_Select3 = "als ganzes Wort"
L.Options_Mapping_MatchWhen_Select4 = "nur am Anfang eines Worts"
L.Options_Mapping_MatchWhen_Select5 = "nur am Ende eines Worts"
L.Options_Mapping_MatchWhen_Select6 = "nur am Anfang oder Ende eines Worts"
L.Options_Mapping_MatchWhen_Select7 = "nie am Anfang und Ende eines Worts"
L.Options_Mapping_SearchText_Name = "Suchtext:" L.Options_Mapping_SearchText_Name = "Suchtext:"
L.Options_Mapping_SearchText_Desc = "Dieser Text wird in der Chateingabe gesucht." L.Options_Mapping_SearchText_Desc = "Dieser Text wird in der Chateingabe gesucht."
L.Options_Mapping_ReplaceText_Name = "Ersetzung:" L.Options_Mapping_ReplaceText_Name = "Ersetzung:"
@ -112,10 +126,6 @@ L.Options_Mapping_Consolidate_Name = "Zusammenfassen aufeinanderfolgender Treffe
L.Options_Mapping_Consolidate_Desc = "Wenn durch die Ersetzung die Zeichenfolge mehrfach hintereinander steht,|nfasse sie zu einem Vorkommen zusammen." L.Options_Mapping_Consolidate_Desc = "Wenn durch die Ersetzung die Zeichenfolge mehrfach hintereinander steht,|nfasse sie zu einem Vorkommen zusammen."
L.Options_Mapping_StopOnMatch_Name = "Anhalten nach Treffer" L.Options_Mapping_StopOnMatch_Name = "Anhalten nach Treffer"
L.Options_Mapping_StopOnMatch_Desc = "F\195\188hrt keine nachfolgenden Ersetzungen mehr durch, wenn dieser Eintrag ein Suchtreffer war." L.Options_Mapping_StopOnMatch_Desc = "F\195\188hrt keine nachfolgenden Ersetzungen mehr durch, wenn dieser Eintrag ein Suchtreffer war."
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_Name = "L\195\182schen"
L.Options_Mapping_Delete_Desc = "L\195\182scht diese Zuweisung." L.Options_Mapping_Delete_Desc = "L\195\182scht diese Zuweisung."
L.Options_Mapping_Delete_ConfirmText="Diese Zuweisung l\195\182schen?" L.Options_Mapping_Delete_ConfirmText="Diese Zuweisung l\195\182schen?"
@ -145,9 +155,10 @@ L.Options_Help_Basics = cYellow("Reihenfolge")
.. "|nDas Zusammenfassen aufeinanderfolgender Treffer vermeidet unsch\195\182ne Wiederholungen, die durch die Ersetzung entstehen k\195\182nnen. " .. "|nDas Zusammenfassen aufeinanderfolgender Treffer vermeidet unsch\195\182ne Wiederholungen, die durch die Ersetzung entstehen k\195\182nnen. "
.. "Die Zusammenfassung wird erst nach der Ersetzung vorgenommen, d.h. am vollst\195\164ndig ersetzten Text f\195\188r jede Zuordnung. " .. "Die Zusammenfassung wird erst nach der Ersetzung vorgenommen, d.h. am vollst\195\164ndig ersetzten Text f\195\188r jede Zuordnung. "
.. "|nMit der Zuordnung " .. cPrefix("\"s\" => \"sch\"") .. " wird aus " .. cPrefix("\"Tasse\" => \"Tasche\"") .. " statt " .. cPrefix("\"Taschsche\"") .. ", " .. "|nMit der Zuordnung " .. cPrefix("\"s\" => \"sch\"") .. " wird aus " .. cPrefix("\"Tasse\" => \"Tasche\"") .. " statt " .. cPrefix("\"Taschsche\"") .. ", "
.. "aber " .. cPrefix("\"schmeissen\" => \"schchmeischen\"") .. ". Solche Randbedingungen beseitigt in der Regel eine weitere Zuordnung wie " .. cPrefix("\"chch\" => \"ch\"") .. "." .. "aber " .. cPrefix("\"schmeissen\" => \"schchmeichen\"") .. ". Solche Randbedingungen beseitigt in der Regel eine weitere Zuordnung wie " .. cPrefix("\"schch\" => \"sch\"") .. "."
.. "|n|n" .. cYellow("Anhalten nach Treffer") .. "|n|n" .. cYellow("Anhalten nach Treffer")
.. "|nEs werden keine weiteren Ersetzungen mehr vorgenommen, wenn die aktuelle Zuordnung zutreffend ist. Alle nachfolgenden Zuordnungen werden dann \195\188bersprungen. Wenn kein Treffer vorliegt, werden die restlichen Zuordnung ganz normal weiter abgearbeitet." .. "|nEs werden keine weiteren Ersetzungen mehr vorgenommen, wenn die aktuelle Zuordnung zutreffend ist. Alle nachfolgenden Zuordnungen werden dann \195\188bersprungen. "
.. "Wenn kein Treffer vorliegt, werden die restlichen Zuordnung ganz normal weiter abgearbeitet."
L.Options_Help_Expert = cYellow("verk\195\188rzende/verl\195\163ngernde Ersetzungen") L.Options_Help_Expert = cYellow("verk\195\188rzende/verl\195\163ngernde Ersetzungen")
.. "|nIst der Ersetzungstext k\195\188rzer als der eigentliche Suchtext, werden die \195\188bersch\195\188\195\159igen Zeichen des Suchtreffers entfernt. " .. "|nIst der Ersetzungstext k\195\188rzer als der eigentliche Suchtext, werden die \195\188bersch\195\188\195\159igen Zeichen des Suchtreffers entfernt. "
.. "Ist der Ersetzungstext l\195\163nger, werden die \195\188brigen Zeichen nach dem Treffer hinten drangehangen. Dabei wird die Gro\195\159- und Kleinschreibung des letzten Zeichens ber\195\188cksichtigt, " .. "Ist der Ersetzungstext l\195\163nger, werden die \195\188brigen Zeichen nach dem Treffer hinten drangehangen. Dabei wird die Gro\195\159- und Kleinschreibung des letzten Zeichens ber\195\188cksichtigt, "
@ -157,39 +168,397 @@ L.Options_Help_Expert = cYellow("verk\195\188rzende/verl\195\163ngernde Ersetzun
.. "|n|n" .. cYellow("Standby") .. "|n|n" .. cYellow("Standby")
.. "|nErsetzungen k\195\182nnen auch nur bei Bedarf durchgef\195\188hrt werden, selbst wenn das Addon oder ein Kanal deaktivert wurde. " .. "|nErsetzungen k\195\182nnen auch nur bei Bedarf durchgef\195\188hrt werden, selbst wenn das Addon oder ein Kanal deaktivert wurde. "
.. "Vor der Eingabe in der Chatbox schreibt man " .. cPrefix("/gri").. " oder " .. cPrefix("/grichelde").. " und optional noch den Zielkanal " .. "Vor der Eingabe in der Chatbox schreibt man " .. cPrefix("/gri").. " oder " .. cPrefix("/grichelde").. " und optional noch den Zielkanal "
.. "z.B. " .. cPrefix("\"/gri /p hallo da dr\195\188ben\"") .. " und alle aktiven Zuordnungen werden ersetzt, selbst wenn der Gruppenkanal oder das Addon nicht aktiv sind." .. "z.B. " .. cPrefix("\"/gri /g hallo leute\"") .. " und alle aktiven Zuordnungen werden ersetzt, selbst wenn der Gildenkanal oder das Addon nicht aktiv sind."
.. "|n|n" .. cYellow("Regul\195\164re Ausdr\195\188cke") .. "|n|n" .. cYellow("Regul\195\164re Ausdr\195\188cke")
.. "|nRegEx sind sehr m\195\163chtige Such- und Ersetzunsgmuster die h\195\163ufig in der Programmierung verwendet werden. Technisch gesehen benutzt das Addon f\195\188r die Suchen des Eingabetextes bereits regul\195\163ren Ausdr\195\188cke. " .. "|nRegEx sind sehr m\195\163chtige Such- und Ersetzunsgmuster die h\195\163ufig in der Programmierung verwendet werden. Generell werden RegEx in den Suchtexten \195\188bernommen, "
.. "Das Eingeben von RegEx als Suchtext ist allerings eine inoffizielle Funktion und hat zwei gro\195\159e Einschr\195\163nkungen: " .. "aber Lua unterst\195\188tzt nicht den vollst\195\163ndigen Umfang von PCRE. Trotzdem funktionieren viele Muster wie Anker bei Zeilenanfang " .. cPrefix("\"^\"") .. " oder Zeilenende " .. cPrefix("\"$\"")
.. "|n1. Leider unterst\195\188tzt Lua nicht den vollst\195\163ndigen Umfang von PCRE. Trotzdem k\195\182nnen einige Muster verwendet werden wie Zeilenanfang " .. cPrefix("\"^\"") .. " " .. ", Gruppen " .. cPrefix("\"(Hallo) (Welt)\"") .. "Zeichenklassen wie Zahlen " .. cPrefix("\"%d\"") .. " oder (negierte) Auswahlen " .. cPrefix("\"[^%p]\"") .. ". "
.."oder Zeilenende " .. cPrefix("\"$\"") .. ", Zeichenklassen wie Zahlen " .. cPrefix("\"%d\"") .. " oder (negierte) Auswahlen " .. cPrefix("\"[^%p]\"") .. ". " .. "Auf Gruppen kann im Ersetzungtext mit %<Nummer> zugegriffen werden" .. cPrefix("\"%2 %1\"") .. "."
.. "|n2. Es werden keine Gruppen im Ersetzungstest unterst\195\188tzt, so da\195\159 Gruppen einfach verloren gehen. Wegen der Gro\195\159- und Kleinschreibung und steigender Komplexit\195\163t ist diese Funktion auch f\195\188r die Zukunft nicht geplant. "
.. "|nIm Beispiel-Reiter gibt es einige Ersetzungen, welche mit regul\195\164ren Ausdr\195\188cke umgesetzt wurden." .. "|nIm Beispiel-Reiter gibt es einige Ersetzungen, welche mit regul\195\164ren Ausdr\195\188cke umgesetzt wurden."
L.Options_Help_Examples_Note = cYellow("Hinweis:") .. " Dieses Addon bef\195\188rwortet nicht und beabsichtig nicht Personen mit (Fremd-)Sprachproblemen |nzu verletztem oder herabzuw\195\188rdigen. Die Verantwortung f\195\188r den Einsatz des Addons obliegt dem Benutzer. |nBitte verwendet die Funktion respektvoll und zur\195\188ckhaltend gegen\195\188ber anderen Mitspielern." L.Options_Help_Examples_Note = cYellow("Hinweis:") .. " Dieses Addon bef\195\188rwortet nicht und beabsichtig nicht Personen mit (Fremd-)Sprachproblemen |nzu verletztem oder herabzuw\195\188rdigen. Die Verantwortung f\195\188r den Einsatz des Addons obliegt dem Benutzer. |nBitte verwendet die Funktion respektvoll und zur\195\188ckhaltend gegen\195\188ber anderen Mitspielern."
L.Options_Help_Examples0_Header = cYellow("Beispiel") L.Options_Help_Examples_Header = cYellow("Beispiel")
L.Options_Help_Examples0_Text = "Bitte ein Beispiel aus der Auswahlbox ausw\195\164hlen." L.Options_Help_Examples_Text = "Bitte ein Beispiel aus der Auswahlbox ausw\195\164hlen."
L.Options_Help_Examples1_Select = "fehlender Unterkiefer" L.Options_Help_Examples_Import_Name = "Importieren"
L.Options_Help_Examples1_Header = cYellow("S und P werden durch Zisch- und Klacklaute ersetzt.") L.Options_Help_Examples_Import_Desc = "Importiert das ausgew\195\164hlte Beispiel in ein neues Profil %s."
L.Options_Help_Examples1_Text = cPrefix("s => ch") .. "|n|n" .. cPrefix("t => ck") L.Options_Help_Examples_Import_ConfirmText = "Wird das Beispiel %s in das neue Profil %s importieren."
L.Options_Help_Examples2_Select = "Trollifizierung"
L.Options_Help_Examples2_Header = cYellow("L\195\164\195\159t euch fast wie ein echter Troll klingen.") L.Options_Help_Examples = {
L.Options_Help_Examples2_Text = cPrefix("%.$ => , maan.") .. "|n|n" .. cPrefix("ir => ia") .. "|n|n" .. cPrefix("ch => ck") .. "|n|n" .. cPrefix("g => ch") .. "|n|n" .. cPrefix("qu => kw") .. "|n|n" .. cPrefix("t => d") {
L.Options_Help_Examples3_Select = "Altmodisch" name = "fehlender Unterkiefer",
L.Options_Help_Examples3_Header = cYellow("Benutzt eine antiquiertere Schreibweise.") desc = cYellow("S und P werden durch Zisch- und Klacklaute ersetzt."),
L.Options_Help_Examples3_Text = cPrefix("ei => ey") .. "|n|n" .. cPrefix("\195\159 => sz") replacements = {
L.Options_Help_Examples4_Select = "Abk\195\188rzungen" replacement_10 = {
L.Options_Help_Examples4_Header = cYellow("Viel sagen, wenig tippen.") order = 10,
L.Options_Help_Examples4_Text = cPrefix("gz => Herzlichen Gl\195\188ckwunsch") .. "|n|n" .. cPrefix("gn8 => Gute Nacht") .. "|n|n" .. cPrefix("afk => Bin mal eben weg (AFK)") .. "|n|n" .. cPrefix("MC => Geschmolzener Kern") searchText = "s",
L.Options_Help_Examples5_Select = "Eigen-, Kose- und Ortsnamen" replaceText = "ch",
L.Options_Help_Examples5_Header = cYellow("Ersetzt Spielernamen, NPCs oder Orte durch andere Ausdr\195\188cke.") exactCase = false,
L.Options_Help_Examples5_Text = "Exakte Gro\195\159- und Kleinschreibung wird empfohlen|n|n" .. cPrefix("Sylvanas => die rachs\195\188chtige Bansheek\195\182nigin") .. "|n|n" .. cPrefix("R\195\188tzkn\195\188bbel => R\195\188tzi") .. "|n|n" .. cPrefix("Unterstadt => Undercity") consolidate = true,
L.Options_Help_Examples6_Select = "Lispeln" matchWhen = 2,
L.Options_Help_Examples6_Header = cYellow("Aussprache von S und Z wird zu einem Zischlaut") stopOnMatch = false,
L.Options_Help_Examples6_Text = cPrefix("sch => ch") .. "|n|n" .. cPrefix("s => fs") .. "|n|n" .. cPrefix("z => ts") .. "|n|n" .. cPrefix("chs => x") },
L.Options_Help_Examples7_Select = "Stottern" replacement_11 = {
L.Options_Help_Examples7_Header = cYellow("Stottern") order = 11,
L.Options_Help_Examples7_Text = "p[%s]-$" searchText = "t",
replaceText = "ck",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "p",
replaceText = "b",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
}
}, {
name = "Trollifizierung",
desc = cYellow("L\195\164\195\159t euch fast wie Vol'jin klingen."),
replacements = {
replacement_10 = {
order = 10,
searchText = "(%w)(%p?)$",
replaceText = "%1, maan%2",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "ir(r?)",
replaceText = "ia",
exactCase = false,
consolidate = false,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "ch",
replaceText = "ck",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "g",
replaceText = "ch",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "qu",
replaceText = "kw",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "t",
replaceText = "d",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_16 = {
order = 16,
searchText = "er",
replaceText = "a",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
}
}, {
name = "Jar Jar Binks",
desc = cYellow("L\195\164\195\159t euch sprechen wie ein ungeschickter Gungan"),
replacements = {
replacement_10 = {
order = 10,
searchText = "ver",
replaceText = "va",
exactCase = false,
consolidate = false,
matchWhen = 4,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "en",
replaceText = "'n",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "er",
replaceText = "a",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "(%w?)ich",
replaceText = "%1ichse",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "(d?m?)ir",
replaceText = "%1ichse",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "du",
replaceText = "du da",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_16 = {
order = 16,
searchText = "er",
replaceText = "erse",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_17 = {
order = 17,
searchText = "sie",
replaceText = "sie da",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_18 = {
order = 18,
searchText = "wir",
replaceText = "wirse",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_19= {
order = 19,
searchText = "ihr",
replaceText = "ihrse",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_20 = {
order = 20,
searchText = "nicht",
replaceText = "nich",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_21 = {
order = 21,
searchText = "die",
replaceText = "de",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
}
}, {
name = "Altmodisch",
desc = cYellow("Benutzt eine antiquiertere Schreibweise."),
replacements = {
replacement_10 = {
order = 10,
searchText = "ei",
replaceText = "ey",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "eu",
replaceText = "oy",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "\195\159",
replaceText = "sz",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
}
}, {
name = "Abk\195\188rzungen",
desc = cYellow("Viel sagen, wenig tippen."),
replacements = {
replacement_10 = {
order = 10,
searchText = "gz",
replaceText = "Herzlichen Gl\195\188ckwunsch",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "gn8",
replaceText = "Gute Nacht",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "afk",
replaceText = "Bin mal kurz weg. (AFK)",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "MC",
replaceText = "Geschmolzener Kern",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
}
}, {
name = "Eigen-, Kose- und Ortsnamen",
desc = cYellow("Ersetzt Spielernamen, NPCs oder Orte durch andere Ausdr\195\188cke."),
replacements = {
replacement_10 = {
order = 10,
searchText = "Sylvanas",
replaceText = "die rachs\195\188chtige Bansheek\195\182nigin",
exactCase = true,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "R\195\188tzkn\195\188bbel",
replaceText = "R\195\188tzi",
exactCase = true,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "Unterstadt",
replaceText = "Undercity",
exactCase = true,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
}
}, {
name = "Lispeln",
desc = cYellow("Aussprache von S und Z wird zu einem Zischlaut"),
replacements = {
replacement_10 = {
order = 10,
searchText = "sch",
replaceText = "ch",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "s",
replaceText = "fs",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "z",
replaceText = "ts",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "chs",
replaceText = "x",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
}
}, {
name = "Stottern",
desc = cYellow("Wiederholt Vokale am Satzanfang"),
replacements = {
replacement_10 = {
order = 10,
searchText = "^([^aeiouy]*)([aeiouy])",
replaceText = "%1%2-%1%2-%1%2",
exactCase = false,
consolidate = true,
matchWhen = 4,
stopOnMatch = false,
},
}
},
}
L.IgnorePattern_Star = "Stern" L.IgnorePattern_Star = "Stern"
L.IgnorePattern_Circle = "Kreis" L.IgnorePattern_Circle = "Kreis"

@ -6,9 +6,9 @@ local Grichelde = _G.Grichelde
local L = LibStub('AceLocale-3.0'):NewLocale(AddonName, 'enUS', true) local L = LibStub('AceLocale-3.0'):NewLocale(AddonName, 'enUS', true)
if not L then return end if not L then return end
local cYellow = Grichelde.functions.cYellow local cYellow = Grichelde.F.cYellow
local cGray = Grichelde.functions.cGray local cGray = Grichelde.F.cGray
local cPrefix = Grichelde.functions.cPrefix local cPrefix = Grichelde.F.cPrefix
-- system messages -- system messages
L.AddonName = "Grichelde" L.AddonName = "Grichelde"
@ -17,6 +17,8 @@ L.AddonLoaded = "%s happily assists you with your spelling disabilities now."
L.AddonUnloaded = "%s patiently waits to support you again when needed." L.AddonUnloaded = "%s patiently waits to support you again when needed."
L.Upgrade_ToVersion = "Upgrade database to version %s." L.Upgrade_ToVersion = "Upgrade database to version %s."
L.Upgrade_Successful = "Upgrade successful." L.Upgrade_Successful = "Upgrade successful."
L.Upgrade_Error = "Upgrade failed!"
L.Downgrade_Detected = "Downgrade detected, %s might not work correctly!"
-- debug -- debug
L.Debug_Options = "Options" L.Debug_Options = "Options"
@ -41,6 +43,7 @@ L.Profiles_Copied = "Settings applied from profile %s."
L.Profiles_Reset = "Profil %s reset." L.Profiles_Reset = "Profil %s reset."
L.Profiles_Invalid = "Invalid profile %s!" L.Profiles_Invalid = "Invalid profile %s!"
L.Profiles_DeleteError = "The active profile cannot be deleted!" L.Profiles_DeleteError = "The active profile cannot be deleted!"
L.Profiles_AlreadyExistsError = "The profile %s already exists!"
-- minimap -- minimap
L.Minimap_Tooltip_Enabled = "%s" L.Minimap_Tooltip_Enabled = "%s"
@ -100,8 +103,19 @@ L.Options_Replacements_Header = "All matches on the lefthand side of the arrow \
L.Options_Mapping_Group_Name = "%s => %s" L.Options_Mapping_Group_Name = "%s => %s"
L.Options_Mapping_Group_Desc = "This lookup will be replaced in activated channels." L.Options_Mapping_Group_Desc = "This lookup will be replaced in activated channels."
L.Options_Mapping_EmptyMapping = "(none)" L.Options_Mapping_EmptyMapping = "(none)"
L.Options_Mapping_Enabled_Name = "active" L.Options_Mapping_MoveUp_Name = "^"
L.Options_Mapping_Enabled_Desc = "This replacement will be processed." L.Options_Mapping_MoveUp_Desc = "move up"
L.Options_Mapping_MoveDown_Name = "v"
L.Options_Mapping_MoveDown_Desc = "move down"
L.Options_Mapping_MatchWhen_Name = "when:"
L.Options_Mapping_MatchWhen_Desc = "Replacement is only done if the search text matches either |nanywhere (<always>), |nif the search text mantches <as a whole word>, |nolny at the <start of each word>, |nor at the <end of each word>, |nor <only at the start or end of each word> but not in between, |nor only in the middle of each word, but <never at start or end of any word>."
L.Options_Mapping_MatchWhen_Select1 = "never (disabled)"
L.Options_Mapping_MatchWhen_Select2 = "always"
L.Options_Mapping_MatchWhen_Select3 = "as a whole word"
L.Options_Mapping_MatchWhen_Select4 = "start of each word"
L.Options_Mapping_MatchWhen_Select5 = "end of each word"
L.Options_Mapping_MatchWhen_Select6 = "only at start and end of each word"
L.Options_Mapping_MatchWhen_Select7 = "never at start or end of any word"
L.Options_Mapping_SearchText_Name = "Search for:" L.Options_Mapping_SearchText_Name = "Search for:"
L.Options_Mapping_SearchText_Desc = "This text is looked up in your chat input box." L.Options_Mapping_SearchText_Desc = "This text is looked up in your chat input box."
L.Options_Mapping_ReplaceText_Name = "Replacement:" L.Options_Mapping_ReplaceText_Name = "Replacement:"
@ -112,10 +126,6 @@ L.Options_Mapping_Consolidate_Name = "consolidate consecutive matches"
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_Consolidate_Desc = "If after the replacement a text sequence is repeated|ndirectly after another, treat them as one occurrence."
L.Options_Mapping_StopOnMatch_Name = "stop on match" L.Options_Mapping_StopOnMatch_Name = "stop on match"
L.Options_Mapping_StopOnMatch_Desc = "Stops looking for any following replacements, when this one matched." L.Options_Mapping_StopOnMatch_Desc = "Stops looking for any following replacements, when this one matched."
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_Name = "Delete"
L.Options_Mapping_Delete_Desc = "Deletes this replacement mapping." L.Options_Mapping_Delete_Desc = "Deletes this replacement mapping."
L.Options_Mapping_Delete_ConfirmText = "Delete this replacement mapping?" L.Options_Mapping_Delete_ConfirmText = "Delete this replacement mapping?"
@ -158,38 +168,440 @@ L.Options_Help_Expert = cYellow("shortening/lengthening replacements")
.. "In the chatbox put " .. cPrefix("/gri") .. " or " .. cPrefix("/grichelde") .. " in front of your typed text, you can also include the target channel, " .. "In the chatbox put " .. cPrefix("/gri") .. " or " .. cPrefix("/grichelde") .. " in front of your typed text, you can also include the target channel, "
.. "i.e. " .. cPrefix("\"/gri /guild hello there\"") .. " and the active replacements are applied even if the guild channel or global switch was disabled." .. "i.e. " .. cPrefix("\"/gri /guild hello there\"") .. " and the active replacements are applied even if the guild channel or global switch was disabled."
.. "|n|n" .. cYellow("Regular Expressions") .. "|n|n" .. cYellow("Regular Expressions")
.. "|nRegex are very powerful search and replacement patterns commonly used in programming. Technically all searches the addon performs on the input text are done with regular expression methods. " .. "|nRegEx are very powerful search and replacement patterns commonly used in programming. Technically all searches the addon performs on the input are done with regular expression methods, "
.. "Entering regex as search text however is an unofficial feature and has two major caveats: " .. "however Lua unfortunately does not support full PCRE syntax and is very limited. Nevertheless some patterns can be used like the anchors start of line " .. cPrefix("\"^\"") .. " or end of line " .. cPrefix("\"$\"")
.. "|n1. Unfornately Lua does not support full PCRE syntax and is very limited. Nethertheless some patterns can be used like start of line " .. cPrefix("\"^\"") .. " or end of line " .. cPrefix("\"$\"") .. ", " .. ", capturing groups " .. cPrefix("\"(hello) (world)\"") .. "character classes like numbers " .. cPrefix("\"%d\"") .. " or (inversed) sets " .. cPrefix("\"[^%p]\"") .. ". "
.. "character classes like numbers " .. cPrefix("\"%d\"") .. " or (inversed) sets " .. cPrefix("\"[^%p]\"") .. ". " .. "Capture groups can be accessed in the replacement text with %<number> like in " .. cPrefix("\"%2 %1\"").. "."
.. "|n2. There is no support for capture groups in the replacement text, so matches get lost. Because of case sensivity and complexity there are no plans to support this." .. "|nAlso there are some mappings using RegEx in the Example section."
.. "|nAnyway, there are some mappings using RegEx in the Example secion."
L.Options_Help_Examples_Note = cYellow("Note:") .. " This addon does not encourange or intend to hurt or to tease people with speaking disabilities or language disorders. The responsibility rest on the user completely. Please use the features with care and respect to other players." L.Options_Help_Examples_Note = cYellow("Note:") .. " This addon does not encourange or intend to hurt or to tease people with speaking disabilities or language disorders. The responsibility rest on the user completely. Please use the features with care and respect to other players."
L.Options_Help_Examples0_Header = cYellow("Example") L.Options_Help_Examples_Header = cYellow("Example")
L.Options_Help_Examples0_Text = "Select an example from the dropdown above." L.Options_Help_Examples_Text = "Select an example from the dropdown above."
L.Options_Help_Examples1_Select = "absent jaw" L.Options_Help_Examples_Import_Name = "Import"
L.Options_Help_Examples1_Header = cYellow("S and P will be replaced by sibilant and clack sounds.") L.Options_Help_Examples_Import_Desc = "Imports the selected example into a new profile."
L.Options_Help_Examples1_Text = cPrefix("s => ch") .. "|n|n" .. cPrefix("t => ck") L.Options_Help_Examples_Import_ConfirmText = "This will import the example %s into the nre profile %s."
L.Options_Help_Examples2_Select = "trollifier"
L.Options_Help_Examples2_Header = cYellow("Almost sound like a real Troll.") L.Options_Help_Examples = {
L.Options_Help_Examples2_Text = cPrefix("%.$ => , mon.") .. "|n|n" .. cPrefix("th => d") .. "|n|n" .. cPrefix("you => ya") .. "|n|n" .. cPrefix("ing => in'") {
L.Options_Help_Examples3_Select = "old-fashioned" name = "absent jaw",
L.Options_Help_Examples3_Header = cYellow("Use an outdate pronounciation.") desc = cYellow("S and P will be replaced by sibilant and clack sounds."),
L.Options_Help_Examples3_Text = cPrefix("oi => oy") .. "|n|n" .. cPrefix("do => doe") .. "|n|n" .. cPrefix("go => goe") .. "|n|n" .. cPrefix("you => thou") .. "|n|n" .. cPrefix("yours => thy") .. "|n|n" .. cPrefix("be => bee") .. "|n|n" .. cPrefix("we => wee") replacements = {
L.Options_Help_Examples4_Select = "abbreviations" replacement_10 = {
L.Options_Help_Examples4_Header = cYellow("Say much, type less.") order = 10,
L.Options_Help_Examples4_Text = cPrefix("gz => Congratulations") .. "|n|n" .. cPrefix("gn8 => Good night") .. "|n|n" .. cPrefix("afk => I'm temporarikly not available (AFK)") .. "|n|n" .. cPrefix("MC => Molten Core") searchText = "s",
L.Options_Help_Examples5_Select = "Proper names" replaceText = "ch",
L.Options_Help_Examples5_Header = cYellow("Replace player names, NPCs or locations.") exactCase = false,
L.Options_Help_Examples5_Text = "exact case is recommended|n|n" .. cPrefix("Sylvanas => the revengeful banshee queen") .. "|n|n" .. cPrefix("Asmongold => Asmon") .. "|n|n" .. cPrefix("Crossroads => X-roads") consolidate = true,
L.Options_Help_Examples6_Select = "lisp" matchWhen = 2,
L.Options_Help_Examples6_Header = cYellow("S and Z will become a sibilant") stopOnMatch = false,
L.Options_Help_Examples6_Text = cPrefix("s => th") .. "|n|n" .. cPrefix("ch => tsh") .. "|n|n" .. cPrefix("z => tsh") .. "|n|n" .. cPrefix("dg => ck") },
L.Options_Help_Examples7_Select = "stammer" replacement_11 = {
L.Options_Help_Examples7_Header = cYellow("stammer") order = 11,
L.Options_Help_Examples7_Text = "p[% s]-$" searchText = "t",
replaceText = "ck",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "p",
replaceText = "b",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
}
}, {
name = "trollifier",
desc = cYellow("Almost sound like Vol'jin."),
replacements = {
replacement_10 = {
order = 10,
searchText = "(%w)(%p?)$",
replaceText = "%1, mon%2",
exactCase = false,
consolidate = false,
matchWhen = 5,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "th",
replaceText = "d",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "what are you",
replaceText = "whatcha",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "your?s?",
replaceText = "ya",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "going to",
replaceText = "gonna",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "want to",
replaceText = "wanna",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_16 = {
order = 16,
searchText = "ing",
replaceText = "in'",
exactCase = false,
consolidate = true,
matchWhen = 5,
stopOnMatch = false,
},
}
}, {
name = "Jar Jar Binks",
desc = cYellow("Lets you sound like a clumsy Gungan"),
replacements = {
replacement_10 = {
order = 10,
searchText = "me",
replaceText = "mesa",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "I am",
replaceText = "Mesa",
exactCase = true,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "I'm",
replaceText = "Mesa",
exactCase = true,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "I",
replaceText = "Me",
exactCase = true,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_14 = {
order = 14,
searchText = "you are",
replaceText = "yousa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_15 = {
order = 15,
searchText = "you're",
replaceText = "yousa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_16 = {
order = 16,
searchText = "your",
replaceText = "yous",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_17 = {
order = 17,
searchText = "(s?)he is",
replaceText = "%1hesa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_18 = {
order = 18,
searchText = "(s?)he's",
replaceText = "%1hesa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_19 = {
order = 19,
searchText = "they",
replaceText = "daysa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_20 = {
order = 20,
searchText = "them",
replaceText = "them-sa",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_21 = {
order = 21,
searchText = "ing",
replaceText = "in'",
exactCase = false,
consolidate = true,
matchWhen = 5,
stopOnMatch = false,
},
replacement_22 = {
order = 22,
searchText = "the",
replaceText = "da",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_23 = {
order = 23,
searchText = "th",
replaceText = "d",
exactCase = false,
consolidate = false,
matchWhen = 2,
stopOnMatch = false,
},
replacement_24 = {
order = 24,
searchText = "yes",
replaceText = "yesa",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
replacement_25 = {
order = 25,
searchText = "oka?y?",
replaceText = "okeeday",
exactCase = false,
consolidate = false,
matchWhen = 3,
stopOnMatch = false,
},
}
}, {
name = "old-fashioned",
desc = cYellow("Use an outdated pronounciation."),
replacements = {
replacement_10 = {
order = 10,
searchText = "oi",
replaceText = "oy",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "([^aeiou]*)([aeiou])",
replaceText = "%1%2e",
exactCase = false,
consolidate = true,
matchWhen = 5,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "yours",
replaceText = "thy",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "youe",
replaceText = "thou",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
}
}, {
name = "abbreviations",
desc = cYellow("Say much, type less."),
replacements = {
replacement_10 = {
order = 10,
searchText = "gz",
replaceText = "Congratulations",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "gn8",
replaceText = "Good night",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "afk",
replaceText = "I'm temporarily unavailable (AFK)",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "MC",
replaceText = "Molten Core",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
}
}, {
name = "Proper names",
desc = cYellow("Replace player names, NPCs or locations."),
replacements = {
replacement_10 = {
order = 10,
searchText = "Sylvanas",
replaceText = "the revengeful banshee queen",
exactCase = true,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "Asmon",
replaceText = "Asmongold",
exactCase = true,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "x%-?roads",
replaceText = "Crossroads",
exactCase = false,
consolidate = true,
matchWhen = 3,
stopOnMatch = false,
},
}
}, {
name = "Lisp",
desc = cYellow("S and Z will become sibilant"),
replacements = {
replacement_10 = {
order = 10,
searchText = "s",
replaceText = "th",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_11 = {
order = 11,
searchText = "ch",
replaceText = "tsh",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_12 = {
order = 12,
searchText = "z",
replaceText = "tsh",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
replacement_13 = {
order = 13,
searchText = "dg",
replaceText = "ck",
exactCase = false,
consolidate = true,
matchWhen = 2,
stopOnMatch = false,
},
}
}, {
name = "stammer",
desc = cYellow("Repeats vowels at the beginning of a sentence"),
replacements = {
replacement_10 = {
order = 10,
searchText = "^([^aeiouy]*)([aeiouy])",
replaceText = "%1%2-%1%2-%1%2",
exactCase = false,
consolidate = true,
matchWhen = 4,
stopOnMatch = false,
},
}
},
}
L.IgnorePattern_Star = "Star" L.IgnorePattern_Star = "Star"
L.IgnorePattern_Circle = "Circle" L.IgnorePattern_Circle = "Circle"

Binary file not shown.

After

Width:  |  Height:  |  Size: 76 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 72 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 73 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 70 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 40 KiB

After

Width:  |  Height:  |  Size: 58 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 121 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 112 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 137 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 129 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 83 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 81 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 208 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 198 KiB

@ -1,6 +1,9 @@
Sample replacement Sample replacement
Text is replaced in the "Say" channel. Text is replaced in the "Say" channel.
Replacements
Define your own set of replacements rules
Replacements Replacements
Create multiple mappings between search text and replacement with additional conditions. Create multiple mappings between search text and replacement with additional conditions.
@ -8,10 +11,13 @@ Channel configuration
individual channel activation individual channel activation
Profiles Profiles
global, per server, per class, per character global, per server, per class, per character or custom
Examples
Templates for various situations, now also with import possibility
New: Examples Help
Templates for various situations Built-in help texts
@ -28,13 +34,13 @@ Profile:
Global, pro Server, pro Klasse, pro Character Global, pro Server, pro Klasse, pro Character
Neu: Beispiele Neu: Beispiele
Vorlagen für verschiedene Zwecke Vorlagen für verschiedene Zwecke, jetzt mit Import-Funktion
Error Reporting Error Reporting
In case of errors please send me the exact error message and please also provide the mapping from "/gri mappings" (see next screen). In case of errors please send me the exact error message and please also provide the mapping from "/gri mappings" (see next screen).
Debug Mappings
In case of errors please send me the exact error message and please also provide the mapping from "/gri mappings".
Debug Mappings
In case of errors please send me the exact error message and please also provide the mapping from "/gri mappings"

Binary file not shown.

Before

Width:  |  Height:  |  Size: 40 KiB

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 114 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 90 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 107 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 87 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 191 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 183 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 94 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 90 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 99 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 96 KiB

Loading…
Cancel
Save