List of commits:
Subject Hash Author Date (UTC)
feat(choir): Add conf spell shortcut default button c8e2c3493896f5ae589c516f742bbafd528140db Vladyslav Bondarenko 2021-11-24 02:28:19
feat(choir)!: Add configuration menu 12319ce873aecea4bfc0addec7c5ac5c1f225237 Vladyslav Bondarenko 2021-11-21 09:26:05
feat(choir): Add unit game tooltip 3ec490e489bb105f5096d5bb3e56814873deec3f Vladyslav Bondarenko 2021-11-20 09:34:12
fix(choir): Raid group frame arrange correctly b592f3f018ee521a821447ccf837aded703ad447 Vladyslav Bondarenko 2021-11-18 22:02:52
feat(choir): Render group role indicator for units 5f3a7c24d3f79035b8faa156355a2d017688134e Vladyslav Bondarenko 2021-11-18 22:02:27
feat(choir): Adjust raid group frame position f99a54133a1761034baca93bfa8814c614191818 Vladyslav Bondarenko 2021-11-18 00:35:35
feat(choir): Add debuff buttons for raid fe66339420a9eb5e4823cc4fe45e0d97b2983073 Vladyslav Bondarenko 2021-11-17 00:55:54
feat(choir): Add texture to health bars ddaeb29d7ecfd95a98bcc85b3f4f7676c7ccd4d7 Vladyslav Bondarenko 2021-11-14 21:40:26
fix(choir)!: Obscure critical error 0f203b4d69f57240e97c66f1ab4510c6ac3e9276 Vladyslav Bondarenko 2021-11-11 12:52:02
feat(choir): Toggle button visibility given roster d193e7b5eb38cb3ac69d74eec1f96e00d90b6098 Vladyslav Bondarenko 2021-11-10 23:46:24
feat(choir)!: Add permanent raid frame 1839c35af4212c09038e972547d6b5893a9ce219 Vladyslav Bondarenko 2021-11-10 16:04:25
feat(choir)!: Employ Clearcasting subset feat 733c81538c3c965f07993fd7ddc482e724121b75 Vladyslav Bondarenko 2021-11-04 22:40:48
fix(choir): Improve shortcut binding keys eb636de6e3f7bada9f0064ffe4db1db3f6433f6c Vladyslav Bondarenko 2021-10-31 18:55:10
fix(choir): Improve choirBindingKey attribute handling 6c5c2214cc1809e5e5e59cd3672da3cca3f2701f Vladyslav Bondarenko 2021-10-31 13:23:24
feat(choir)!: Add spell shortcut prototype d34f22a6983ffc41122acb40d22e3cb29c208a3c Vladyslav Bondarenko 2021-10-31 12:39:39
feat(choir)!: Add spell effects on unit d581df9fce342709267a3221dead4d00b9d14319 Vladyslav Bondarenko 2021-10-31 10:49:19
feat(choir)!: Close spoiler by pressing Esc 22d7370011ac45d25a410a3f569183d0cc9fb232 Vladyslav Bondarenko 2021-10-29 16:10:08
feat(choir)!: Range indicator 931a0510b562986ec76dc22f329f4af4ed723cdf Vladyslav Bondarenko 2021-10-29 10:40:46
fix(choir)!: Health bar in combat a6622578dd5a1b4e4babf699a4cf1e4eb6bb70d6 Vladyslav Bondarenko 2021-10-28 07:42:57
feat(choir)!: Decorate unit buttons 4dc5ed44a9519b275f4256cfe4281110b7a94c9a Vladyslav Bondarenko 2021-10-25 21:20:56
Commit c8e2c3493896f5ae589c516f742bbafd528140db - feat(choir): Add conf spell shortcut default button
Interface options menu features a framework for resetting the add-on
configuration to defaults by clicking a specific button labeled default.
This commit takes advantage of the feature and makes the default button
work as would be expected.

Additionally, refresh the configuration by
using hooks provided by the framework. That way the configuration is
updated in a predictable manner.

Also split the initialization logic sligthly to improve maintainability.

Factory functions in this commit specifically are used to hopefully prevent
breaking modifications to the add-on.
Author: Vladyslav Bondarenko
Author date (UTC): 2021-11-24 02:28
Committer name: Vladyslav Bondarenko
Committer date (UTC): 2021-11-24 02:28
Parent(s): 04ebeb03c078f9cf6bf1801bec19bc7aa9213546
Signer:
Signing key:
Signing status: N
Tree: f778857748bb7d50c3e3b23776504e7cb8c927cd
File Lines added Lines deleted
choir.lua 187 72
File choir.lua changed (mode: 100644) (index 16a11cc..ba0e991)
... ... local function initRaidFrame(rootFrame, spoilerHolder)
1089 1089 return createRaidFrame(rootFrame, spoilerHolder) return createRaidFrame(rootFrame, spoilerHolder)
1090 1090 end end
1091 1091
1092 local function applyConfBindingKey(bindingKeyFrame)
1093 assert (bindingKeyFrame ~= nil)
1092 local function readConfShortcutSpellNameList(spellEditBoxList)
1093 local maxShortcutQuantityPerUnit = 5
1094
1095 assert (spellEditBoxList ~= nil)
1096 assert ('table' == type(spellEditBoxList))
1097 assert (maxShortcutQuantityPerUnit == #spellEditBoxList)
1098
1099 local spellNameList = {}
1100 local w = 0
1101 while (w < #spellEditBoxList) do
1102 w = w + 1
1103
1104 local inputBox = spellEditBoxList[w]
1105 assert (inputBox ~= nil)
1106
1107 local spellName = inputBox:GetText()
1108 assert (spellName ~= nil)
1109 assert ('string' == type(spellName))
1110 spellName = strtrim(spellName)
1111 assert (string.len(spellName) >= 2 and string.len(spellName) <= 144)
1112
1113 if not GetSpellInfo(spellName) then
1114 trace('could not find spell by the name of "' .. spellName .. '"')
1115 end
1094 1116
1117 spellNameList[w] = spellName
1118 end
1119
1120 assert (spellNameList ~= nil)
1121 assert ('table' == type(spellNameList))
1122 assert (maxShortcutQuantityPerUnit == #spellNameList)
1123
1124 return spellNameList
1125 end
1126
1127 local function readConfShortcutBindingKeyMap(shortcutEditBoxMap)
1095 1128 local maxUnitQuantityPerSpoiler = 5 local maxUnitQuantityPerSpoiler = 5
1096 1129 local maxShortcutQuantityPerUnit = 5 local maxShortcutQuantityPerUnit = 5
1097 1130
1098 local shortcutEditBoxMap = bindingKeyFrame.shortcutEditBoxMap
1099 1131 assert (shortcutEditBoxMap ~= nil) assert (shortcutEditBoxMap ~= nil)
1100 1132 assert ('table' == type(shortcutEditBoxMap)) assert ('table' == type(shortcutEditBoxMap))
1101 1133 assert (maxUnitQuantityPerSpoiler == #shortcutEditBoxMap) assert (maxUnitQuantityPerSpoiler == #shortcutEditBoxMap)
1102 1134
1103 local spellEditBoxList = bindingKeyFrame.spellEditBoxList
1104 assert (spellEditBoxList ~= nil)
1105 assert ('table' == type(spellEditBoxList))
1106 assert (maxShortcutQuantityPerUnit == #spellEditBoxList)
1107
1108 1135 local keyMap = {} local keyMap = {}
1109 1136 local p = 0 local p = 0
1110 1137 while (p < maxUnitQuantityPerSpoiler) do while (p < maxUnitQuantityPerSpoiler) do
 
... ... local function applyConfBindingKey(bindingKeyFrame)
1133 1160 end end
1134 1161 keyMap[p] = bindingList keyMap[p] = bindingList
1135 1162 end end
1136 ChoirShortcutBindingKeyMap = keyMap
1137 1163
1138 local spellNameList = {}
1139 local w = 0
1140 while (w < #spellEditBoxList) do
1141 w = w + 1
1164 assert (keyMap ~= nil)
1165 assert ('table' == type(keyMap))
1166 assert (maxUnitQuantityPerSpoiler == #keyMap)
1142 1167
1143 local inputBox = spellEditBoxList[w]
1144 assert (inputBox ~= nil)
1168 return keyMap
1169 end
1145 1170
1146 local spellName = inputBox:GetText()
1147 assert (spellName ~= nil)
1148 assert ('string' == type(spellName))
1149 spellName = strtrim(spellName)
1150 assert (string.len(spellName) >= 2 and string.len(spellName) <= 144)
1171 local function applyConfSpellShortcutFactory(shortcutEditBoxMap, spellEditBoxList)
1172 local maxUnitQuantityPerSpoiler = 5
1173 local maxShortcutQuantityPerUnit = 5
1174
1175 assert (shortcutEditBoxMap ~= nil)
1176 assert ('table' == type(shortcutEditBoxMap))
1177 assert (maxUnitQuantityPerSpoiler == #shortcutEditBoxMap)
1178
1179 assert (spellEditBoxList ~= nil)
1180 assert ('table' == type(spellEditBoxList))
1181 assert (maxShortcutQuantityPerUnit == #spellEditBoxList)
1182
1183 return function(bindingKeyFrame)
1184 assert (bindingKeyFrame ~= nil)
1185
1186 local keyMap = readConfShortcutBindingKeyMap(shortcutEditBoxMap)
1187 if keyMap then
1188 assert (keyMap ~= nil)
1189 assert ('table' == type(keyMap))
1190 assert (maxUnitQuantityPerSpoiler == #keyMap)
1191
1192 ChoirShortcutBindingKeyMap = keyMap
1193 end
1194
1195 local spellNameList = readConfShortcutSpellNameList(spellEditBoxList)
1196 if spellNameList then
1197 assert (spellNameList ~= nil)
1198 assert ('table' == type(spellNameList))
1199 assert (maxShortcutQuantityPerUnit == #spellNameList)
1200
1201 ChoirShortcutSpellNameList = spellNameList
1202 else
1203 trace('could not read shortcut spell name list from interface options')
1204 end
1205
1206 --[[ NOTE Reload the GUI to force update override bindings on secure spell buttons. ]]--
1207 ReloadUI()
1151 1208
1152 spellNameList[w] = spellName
1153 1209 end end
1154 ChoirShortcutSpellNameList = spellNameList
1210 end
1211
1212 local function cancelConfSpellShortcutFactory(shortcutEditBoxMap, spellEditBoxList)
1213 local maxUnitQuantityPerSpoiler = 5
1214 local maxShortcutQuantityPerUnit = 5
1215
1216 assert (shortcutEditBoxMap ~= nil)
1217 assert ('table' == type(shortcutEditBoxMap))
1218 assert (maxUnitQuantityPerSpoiler == #shortcutEditBoxMap)
1219
1220 assert (spellEditBoxList ~= nil)
1221 assert ('table' == type(spellEditBoxList))
1222 assert (maxShortcutQuantityPerUnit == #spellEditBoxList)
1223
1224 return function(bindingKeyFrame)
1225 assert (bindingKeyFrame ~= nil)
1226
1227 local spellList = getShortcutSpellNameList()
1228 assert (spellList ~= nil)
1229 assert (maxShortcutQuantityPerUnit == #spellList)
1230
1231 local i = 0
1232 while (i < maxUnitQuantityPerSpoiler) do
1233 i = i + 1
1234
1235 local spellEditBox = spellEditBoxList[i]
1236 assert (spellEditBox ~= nil)
1237
1238 spellEditBox:SetText(spellList[i])
1239 spellEditBox:SetCursorPosition(0)
1240
1241 local j = 0
1242 while (j < maxShortcutQuantityPerUnit) do
1243 j = j + 1
1244 local editBox = shortcutEditBoxMap[i][j]
1245
1246 assert (editBox ~= nil)
1155 1247
1156 --[[ NOTE Reload the GUI to force update override bindings on secure spell buttons. ]]--
1157 ReloadUI()
1248 local binding = getShortcutBindingKeySuggestion(i, j)
1249 editBox:SetText(binding)
1250 --[[ WARNING Reset cursor position on every refresh to make sure updated text is visible. ]]--
1251 editBox:SetCursorPosition(0)
1252 end
1253 end
1254 end
1158 1255 end end
1159 1256
1160 local function initConf(rootFrame)
1161 assert (rootFrame ~= nil)
1257 local function initConfSpellShortcut(confFrame)
1258 assert (confFrame ~= nil)
1162 1259
1163 1260 local maxUnitQuantityPerSpoiler = 5 local maxUnitQuantityPerSpoiler = 5
1164 1261 local maxShortcutQuantityPerUnit = 5 local maxShortcutQuantityPerUnit = 5
 
... ... local function initConf(rootFrame)
1177 1274 ChoirShortcutBindingKeyMap = getDefaultShortcutKeyBindingMap() ChoirShortcutBindingKeyMap = getDefaultShortcutKeyBindingMap()
1178 1275 end end
1179 1276
1180 local spellList = getShortcutSpellNameList()
1181 assert (spellList ~= nil)
1182 assert (maxShortcutQuantityPerUnit == #spellList)
1183
1184 local confFrame = CreateFrame('FRAME', 'ChoirConfFrame', rootFrame)
1185 confFrame.name = GetAddOnMetadata('Choir', 'Title') or 'Choir'
1186
1187 local h1 = confFrame:CreateFontString()
1188 h1:SetJustifyH('LEFT')
1189 h1:SetJustifyV('TOP')
1190 h1:SetFontObject(GameFontNormalLarge)
1191 h1:SetSize(144, 24)
1192 h1:SetPoint('TOPLEFT', 16, -16)
1193 h1:SetText(confFrame.name .. '-' .. (GetAddOnMetadata('Choir', 'Version') or '0'))
1194
1195 local p1 = confFrame:CreateFontString()
1196 p1:SetJustifyH('LEFT')
1197 p1:SetJustifyV('TOP')
1198 p1:SetFontObject(GameFontWhite)
1199 p1:SetSize(386, 24 * 12)
1200 p1:SetPoint('TOPLEFT', 16, -16 - h1:GetHeight())
1201 p1:SetText('Choir add-on enhances targeting, raid frames and spell buttons. ' ..
1202 'The main purpose of the add-on is to allow the user to target units and ' ..
1203 'to cast spells with a combination of key presses. ' ..
1204 'It is expected to be used only by healers. ' ..
1205 'It is intended as an alternative to mouseover macros.\n\n' ..
1206 'For example, to target a unit with a combination of key presses do the following. ' ..
1207 'First, bind a key to player party spoiler in the native key bindings menu. ' ..
1208 'Then, join a party. Finally, press the hot key that was bound to the player party. ' ..
1209 'A spoiler that is a contextual menu will open in the middle of the screen. ' ..
1210 'Every button under the spoiler corresponds to a party member. ' ..
1211 'To target a party member, press the key that is bound to the unit button. ' ..
1212 'To learn what key is bound to the unit button, ' ..
1213 'read the hint on the unit button itself, enclosed in the square brackets.')
1214
1215
1216 local bindingKeyFrame = CreateFrame('FRAME', 'ChoirConfBindingKeyFrame', confFrame)
1277 local bindingKeyFrame = CreateFrame('FRAME', 'ChoirConfSpellShortcutFrame', confFrame)
1217 1278 bindingKeyFrame.name = 'Spell shortcut' bindingKeyFrame.name = 'Spell shortcut'
1218 1279 bindingKeyFrame.parent = confFrame.name bindingKeyFrame.parent = confFrame.name
1219 1280
 
... ... local function initConf(rootFrame)
1226 1287 while (i < maxUnitQuantityPerSpoiler) do while (i < maxUnitQuantityPerSpoiler) do
1227 1288 i = i + 1 i = i + 1
1228 1289
1229 local unitLabel = bindingKeyFrame:CreateFontString()
1290 local unitLabel = bindingKeyFrame:CreateFontString(bindingKeyFrame:GetName() .. 'UnitText' .. tostring(i))
1230 1291 unitLabel:SetFontObject(GameFontNormal) unitLabel:SetFontObject(GameFontNormal)
1231 1292 unitLabel:SetSize(marginLeft - padding, 24) unitLabel:SetSize(marginLeft - padding, 24)
1232 1293 unitLabel:SetText('Unit ' .. tostring(i)) unitLabel:SetText('Unit ' .. tostring(i))
 
... ... local function initConf(rootFrame)
1245 1306 local editBoxY = marginBottom + (i - 1) * editBox:GetHeight() local editBoxY = marginBottom + (i - 1) * editBox:GetHeight()
1246 1307 editBox:SetPoint('BOTTOMLEFT', editBoxX, editBoxY) editBox:SetPoint('BOTTOMLEFT', editBoxX, editBoxY)
1247 1308
1248 editBox:SetText(getShortcutBindingKeySuggestion(i, j))
1249
1250 1309 editBox:SetAutoFocus(false) editBox:SetAutoFocus(false)
1251 1310 editBox:SetCursorPosition(0) editBox:SetCursorPosition(0)
1252 1311
 
... ... local function initConf(rootFrame)
1260 1319 local spellEditBoxY = marginBottom + 24 * 5 local spellEditBoxY = marginBottom + 24 * 5
1261 1320 spellEditBox:SetPoint('BOTTOMLEFT', spellEditBoxX, spellEditBoxY) spellEditBox:SetPoint('BOTTOMLEFT', spellEditBoxX, spellEditBoxY)
1262 1321
1263 spellEditBox:SetText(spellList[i])
1264
1265 1322 spellEditBox:SetAutoFocus(false) spellEditBox:SetAutoFocus(false)
1266 1323 spellEditBox:SetCursorPosition(0) spellEditBox:SetCursorPosition(0)
1267 1324
 
... ... local function initConf(rootFrame)
1269 1326 spellEditBoxList[i] = spellEditBox spellEditBoxList[i] = spellEditBox
1270 1327 end end
1271 1328
1272 local header = bindingKeyFrame:CreateFontString()
1329 local header = bindingKeyFrame:CreateFontString(bindingKeyFrame:GetName() .. 'HeaderText')
1273 1330 header:SetJustifyH('LEFT') header:SetJustifyH('LEFT')
1274 1331 header:SetJustifyV('TOP') header:SetJustifyV('TOP')
1275 1332 header:SetFontObject(GameFontNormalLarge) header:SetFontObject(GameFontNormalLarge)
 
... ... local function initConf(rootFrame)
1277 1334 header:SetPoint('TOPLEFT', 16, -16) header:SetPoint('TOPLEFT', 16, -16)
1278 1335 header:SetText(bindingKeyFrame.name) header:SetText(bindingKeyFrame.name)
1279 1336
1280 local description = bindingKeyFrame:CreateFontString()
1337 local description = bindingKeyFrame:CreateFontString(bindingKeyFrame:GetName() .. 'ParagraphText')
1281 1338 description:SetJustifyH('LEFT') description:SetJustifyH('LEFT')
1282 1339 description:SetJustifyV('TOP') description:SetJustifyV('TOP')
1283 1340 description:SetFontObject(GameFontWhite) description:SetFontObject(GameFontWhite)
 
... ... local function initConf(rootFrame)
1294 1351
1295 1352 bindingKeyFrame.shortcutEditBoxMap = shortcutEditBoxMap bindingKeyFrame.shortcutEditBoxMap = shortcutEditBoxMap
1296 1353 bindingKeyFrame.spellEditBoxList = spellEditBoxList bindingKeyFrame.spellEditBoxList = spellEditBoxList
1297 bindingKeyFrame.okay = applyConfBindingKey
1354
1355 local applyConfSpellShortcut = applyConfSpellShortcutFactory(shortcutEditBoxMap, spellEditBoxList)
1356 assert (applyConfSpellShortcut ~= nil)
1357
1358 local cancelConfSpellShortcut = cancelConfSpellShortcutFactory(shortcutEditBoxMap, spellEditBoxList)
1359 assert (cancelConfSpellShortcut ~= nil)
1360
1361 bindingKeyFrame.okay = applyConfSpellShortcut
1362 bindingKeyFrame.cancel = cancelConfSpellShortcut
1363 bindingKeyFrame.refresh = function(self)
1364 cancelConfSpellShortcut(self)
1365 end
1366 bindingKeyFrame.default = function()
1367 ChoirShortcutSpellNameList = getDefaultShortcutSpellNameList()
1368 ChoirShortcutBindingKeyMap = getDefaultShortcutKeyBindingMap()
1369 ReloadUI()
1370 --[[ NOTE Refresh callback is executed implicitly here. ]]--
1371 end
1372
1373 return bindingKeyFrame
1374 end
1375
1376 local function initConf(rootFrame)
1377 assert (rootFrame ~= nil)
1378
1379 local confFrame = CreateFrame('FRAME', 'ChoirConfFrame', rootFrame)
1380 confFrame.name = GetAddOnMetadata('Choir', 'Title') or 'Choir'
1381
1382 local h1 = confFrame:CreateFontString()
1383 h1:SetJustifyH('LEFT')
1384 h1:SetJustifyV('TOP')
1385 h1:SetFontObject(GameFontNormalLarge)
1386 h1:SetSize(144, 24)
1387 h1:SetPoint('TOPLEFT', 16, -16)
1388 h1:SetText(confFrame.name .. '-' .. (GetAddOnMetadata('Choir', 'Version') or '0'))
1389
1390 local p1 = confFrame:CreateFontString()
1391 p1:SetJustifyH('LEFT')
1392 p1:SetJustifyV('TOP')
1393 p1:SetFontObject(GameFontWhite)
1394 p1:SetSize(386, 24 * 12)
1395 p1:SetPoint('TOPLEFT', 16, -16 - h1:GetHeight())
1396 p1:SetText('Choir add-on enhances targeting, raid frames and spell buttons. ' ..
1397 'The main purpose of the add-on is to allow the user to target units and ' ..
1398 'to cast spells with a combination of key presses. ' ..
1399 'It is expected to be used only by healers. ' ..
1400 'It is intended as an alternative to mouseover macros.\n\n' ..
1401 'For example, to target a unit with a combination of key presses do the following. ' ..
1402 'First, bind a key to player party spoiler in the native key bindings menu. ' ..
1403 'Then, join a party. Finally, press the hot key that was bound to the player party. ' ..
1404 'A spoiler that is a contextual menu will open in the middle of the screen. ' ..
1405 'Every button under the spoiler corresponds to a party member. ' ..
1406 'To target a party member, press the key that is bound to the unit button. ' ..
1407 'To learn what key is bound to the unit button, ' ..
1408 'read the hint on the unit button itself, enclosed in the square brackets.')
1409
1298 1410
1299 1411 InterfaceOptions_AddCategory(confFrame) InterfaceOptions_AddCategory(confFrame)
1412
1413 local bindingKeyFrame = initConfSpellShortcut(confFrame)
1414 assert (bindingKeyFrame ~= nil)
1300 1415 InterfaceOptions_AddCategory(bindingKeyFrame) InterfaceOptions_AddCategory(bindingKeyFrame)
1301 1416
1302 return confFrame
1417 return confFrame, bindingKeyFrame
1303 1418 end end
1304 1419
1305 1420 local function init(rootFrame) local function init(rootFrame)
Hints:
Before first commit, do not forget to setup your git environment:
git config --global user.name "your_name_here"
git config --global user.email "your@email_here"

Clone this repository using HTTP(S):
git clone https://rocketgit.com/user/vrtc/wowaddons

Clone this repository using ssh (do not forget to upload a key first):
git clone ssh://rocketgit@ssh.rocketgit.com/user/vrtc/wowaddons

Clone this repository using git:
git clone git://git.rocketgit.com/user/vrtc/wowaddons

You are allowed to anonymously push to this repository.
This means that your pushed commits will automatically be transformed into a merge request:
... clone the repository ...
... make some changes and some commits ...
git push origin main