Nenue@70
|
1 -- SkeletonKey
|
Nenue@70
|
2 -- ActionTemplates.lua
|
Nenue@70
|
3 -- Created: 7/29/2016 9:14 PM
|
Nenue@70
|
4 -- %file-revision%
|
Nenue@70
|
5 -- Code dealing with the implementation of action hotkeys
|
Nenue@70
|
6
|
Nenue@70
|
7 local tostring, tonumber, pairs, ipairs = tostring, tonumber, pairs, ipairs
|
Nenue@70
|
8 local unpack, SetBinding = unpack, SetBinding
|
Nenue@70
|
9 local tinsert, tContains, select, wipe = tinsert, tContains, select, table.wipe
|
Nenue@70
|
10 local GetSpellBookItemInfo, GetSpellBookItemName, GetSpellInfo = GetSpellBookItemInfo, GetSpellBookItemName, GetSpellInfo
|
Nenue@70
|
11 local GetSpecialization, GetSpecializationInfo, IsPassiveSpell, IsTalentSpell = GetSpecialization, GetSpecializationInfo, IsPassiveSpell, IsTalentSpell
|
Nenue@70
|
12 local PetHasSpellbook, PetHasActionBar, GetPetActionInfo, HasPetSpells = PetHasSpellbook, PetHasActionBar, GetPetActionInfo, HasPetSpells
|
Nenue@70
|
13 local GetProfessions, GetProfessionInfo, GetTalentInfo = GetProfessions, GetProfessionInfo, GetTalentInfo
|
Nenue@70
|
14 local GetNumBindings, GetBinding = GetNumBindings, GetBinding
|
Nenue@70
|
15
|
Nenue@70
|
16 local _, kb = ...
|
Nenue@70
|
17 local print = (DEVIAN_PNAME == 'SkeletonKey') and function(...) print('SK', ...) end or nop
|
Nenue@70
|
18 local cprint = (DEVIAN_PNAME == 'SkeletonKey') and function(...) _G.print('Cfg', ...) end or nop
|
Nenue@70
|
19
|
Nenue@70
|
20 local CLICK_KEYBINDER_MACRO = "CLICK SkeletonKeyMacro:"
|
Nenue@70
|
21 local CLICK_KEYBINDER_KEY = "CLICK SkeletonKeyKey:"
|
Nenue@70
|
22 local PET_BASIC_SUBTEXT = 'Basic Attack'
|
Nenue@70
|
23 local PET_SPECIAL_SUBTEXT = 'Special Ability'
|
Nenue@70
|
24 local PETACTION_SCRIPT = {
|
Nenue@70
|
25 [PET_ACTION_MOVE_TO] = {'pet_move_to', SLASH_PET_MOVE_TO1},
|
Nenue@70
|
26 [PET_ACTION_ATTACK] = {'pet_attack', SLASH_PET_ATTACK1},
|
Nenue@70
|
27 [PET_ACTION_FOLLOW] = {'pet_follow', SLASH_PET_FOLLOW1},
|
Nenue@70
|
28 [PET_ACTION_WAIT] = {'pet_stay', SLASH_PET_STAY1 },
|
Nenue@70
|
29 [PET_MODE_AGGRESSIVE] = {'pet_aggressive', SLASH_PET_AGGRESSIVE1 },
|
Nenue@70
|
30 [PET_MODE_DEFENSIVE] = { 'pet_defensive', SLASH_PET_DEFENSIVE1},
|
Nenue@70
|
31 [PET_MODE_PASSIVE] = { 'pet_passive', SLASH_PET_PASSIVE1},
|
Nenue@70
|
32 [PET_MODE_ASSIST] = {'pet_assist', SLASH_PET_ASSIST1},
|
Nenue@70
|
33 }
|
Nenue@70
|
34 local SECONDARY_PROFESSIONS = {
|
Nenue@70
|
35 [5] = 3,
|
Nenue@70
|
36 [7] = 4,
|
Nenue@70
|
37 [9] = 5,
|
Nenue@70
|
38 [10] = 6
|
Nenue@70
|
39 }
|
Nenue@70
|
40 local petSpellCache,petSubtextCache
|
Nenue@70
|
41 local SUMMON_RANDOM_FAVORITE_MOUNT_SPELL = 150544
|
Nenue@70
|
42
|
Nenue@70
|
43 --kb.ChangedBindings = {}
|
Nenue@70
|
44 --kb.ActionTypes = {}
|
Nenue@70
|
45
|
Nenue@70
|
46 local atype = kb.ActionTypes
|
Nenue@70
|
47
|
Nenue@70
|
48 --- Caps Lock
|
Nenue@70
|
49 atype['mount'] = function(id, name)
|
Nenue@70
|
50 if id == SUMMON_RANDOM_FAVORITE_MOUNT_SPELL then
|
Nenue@70
|
51 return CLICK_KEYBINDER_MACRO, 'mount_random', "/script C_MountJournal.SummonByID(0)", SkeletonKeyMacro
|
Nenue@70
|
52 else
|
Nenue@70
|
53 return CLICK_KEYBINDER_MACRO, 'mount_'..id, "/script C_MountJournal.SummonByID("..id..")", SkeletonKeyMacro
|
Nenue@70
|
54 end
|
Nenue@70
|
55 end
|
Nenue@70
|
56
|
Nenue@70
|
57 atype['macro'] = function(id, name)
|
Nenue@70
|
58 local _, _, text = GetMacroInfo(id)
|
Nenue@70
|
59 return CLICK_KEYBINDER_MACRO, 'macro_' .. tostring(name), name, SkeletonKeyMacro
|
Nenue@70
|
60 end
|
Nenue@70
|
61
|
Nenue@70
|
62 atype['equipset'] = function(id, name)
|
Nenue@70
|
63 return CLICK_KEYBINDER_MACRO, 'equipset_'..tostring(name), "/script UseEquipmentSet("..tostring(id)..")", SkeletonKeyMacro
|
Nenue@70
|
64 end
|
Nenue@70
|
65
|
Nenue@70
|
66 atype['spell'] = function(id, name)
|
Nenue@70
|
67 local attributeName = name
|
Nenue@70
|
68 if kb.ProfessionCache[id] then
|
Nenue@70
|
69 attributeName = "profession_".. kb.ProfessionCache[id].dynamicIndex .. '_' .. kb.ProfessionCache[id].dynamicSubIndex
|
Nenue@70
|
70 end
|
Nenue@70
|
71 return CLICK_KEYBINDER_KEY, attributeName, name, SkeletonKeyKey
|
Nenue@70
|
72 end
|
Nenue@70
|
73
|
Nenue@70
|
74 atype['petaction'] = function(_, name)
|
Nenue@70
|
75 -- ID doesn't exist for basic commands, even though they can be picked up
|
Nenue@70
|
76 local attributeName, attributeValue = "petaction_" .. tostring(name), "/cast "..tostring(name)
|
Nenue@70
|
77
|
Nenue@70
|
78 if not petSpellCache then
|
Nenue@70
|
79 kb.UpdatePetInfo()
|
Nenue@70
|
80 end
|
Nenue@70
|
81 -- Compose a multi-macro for subtext abilities
|
Nenue@70
|
82 if petSpellCache[name] then
|
Nenue@70
|
83 attributeValue = ""
|
Nenue@70
|
84 for spellName, enabled in pairs(petSubtextCache[petSpellCache[name]]) do
|
Nenue@70
|
85 attributeValue = attributeValue .. "/cast " .. spellName .. "\n"
|
Nenue@70
|
86 end
|
Nenue@70
|
87 end
|
Nenue@70
|
88
|
Nenue@70
|
89 if PETACTION_SCRIPT[name] then
|
Nenue@70
|
90 attributeName, attributeValue = unpack(PETACTION_SCRIPT[name])
|
Nenue@70
|
91 elseif kb.PetCache.special[name] then
|
Nenue@70
|
92 attributeName = "petaction_"..kb.PetCache.special[name][3].."_" .. tonumber(kb.PetCache.special[name][6])
|
Nenue@70
|
93 end
|
Nenue@70
|
94 return CLICK_KEYBINDER_MACRO, attributeName, attributeValue, SkeletonKeyMacro
|
Nenue@70
|
95 end
|
Nenue@70
|
96
|
Nenue@70
|
97 atype['battlepet'] = function(id, name)
|
Nenue@70
|
98 return CLICK_KEYBINDER_MACRO, 'battlepet_' .. tostring(name), SLASH_SUMMON_BATTLE_PET1 .. " " .. tostring(name), SkeletonKeyMacro
|
Nenue@70
|
99 end
|
Nenue@70
|
100
|
Nenue@70
|
101 atype['item'] = function(id, name)
|
Nenue@75
|
102 return CLICK_KEYBINDER_KEY, tostring(name), tostring(name), SkeletonKeyKey
|
Nenue@70
|
103 end
|
Nenue@70
|
104
|
Nenue@70
|
105
|
Nenue@70
|
106 --- Resolves the SecureActionButton attribute names used for the given action
|
Nenue@70
|
107 kb.RegisterAction = function(actionType, id, name)
|
Nenue@70
|
108 assert(atype[actionType], 'Missing actionType handler for `'..tostring(actionType)..'`')
|
Nenue@74
|
109 local prefix, attributeName, attributeValue, button = atype[actionType](id, name)
|
Nenue@74
|
110 local command = prefix .. attributeName
|
Nenue@74
|
111 return attributeName, attributeValue, command, prefix, button
|
Nenue@70
|
112 end
|
Nenue@70
|
113
|
Nenue@70
|
114
|
Nenue@74
|
115 local spells = {}
|
Nenue@74
|
116 local SkeletonKey_GetGenericSpell = function(spellName, spellID, icon)
|
Nenue@74
|
117 if not spells[spellID] then
|
Nenue@74
|
118 spells[spellID] = {}
|
Nenue@74
|
119 spells[spellID].actionType = 'spell'
|
Nenue@74
|
120 spells[spellID].actionID = spellID
|
Nenue@74
|
121 spells[spellID].actionName = spellName
|
Nenue@74
|
122 spells[spellID].iconPath = icon
|
Nenue@74
|
123 spells[spellID].statusText = '|cFFBBBBBBSpell|r'
|
Nenue@74
|
124 spells[spellID].dynamicType = nil
|
Nenue@74
|
125 end
|
Nenue@74
|
126 return spells[spellID]
|
Nenue@74
|
127 end
|
Nenue@74
|
128
|
Nenue@74
|
129 -- tries to resolve spells from talent overrides/profession book/etc
|
Nenue@74
|
130 local dynamicTypes = {['profession'] = 'ProfessionCache', ['talent'] = 'TalentCache', ['petaction'] = 'PetInfoCache'}
|
Nenue@74
|
131 kb.ResolveSpellSlot = function(self)
|
Nenue@74
|
132 local spellName, spellID, command, icon = self.actionName, self.actionID, self.command, self.iconPath
|
Nenue@74
|
133 --print(' In:', spellName, spellID, command)
|
Nenue@74
|
134 --print(GetSpellInfo(spellName or spellID))
|
Nenue@74
|
135 local internalName, _, internalIcon, _, _, _, _ = GetSpellInfo(spellName or spellID)
|
Nenue@74
|
136 local isAvailable = internalName and true
|
Nenue@74
|
137
|
Nenue@74
|
138 if internalName and (internalName ~= spellName) then
|
Nenue@74
|
139 -- it's a binding for the originating spell, leave it as is
|
Nenue@74
|
140 print(' |cFFFF4400spell is an override(', internalName, '~=', spellName,') leave the name info alone')
|
Nenue@74
|
141 self.statusText = '|cFFFFFF00Spell|r'
|
Nenue@74
|
142 self.isAvailable = true
|
Nenue@74
|
143 return
|
Nenue@74
|
144 end
|
Nenue@74
|
145
|
Nenue@74
|
146 -- let's us match spells replaced by talents
|
Nenue@74
|
147 local info = kb.DynamicSpells[internalName or spellName]
|
Nenue@74
|
148 if not info then
|
Nenue@74
|
149 local dynamicType, dynamicIndex, dynamicSubIndex = command:match("(%a+)_(%S+)_(%S+)")
|
Nenue@74
|
150 if kb.DynamicSpells[dynamicType] then
|
Nenue@74
|
151 print('|cFFFF4400resolving dynamic type index:', internalName, spellName, command)
|
Nenue@74
|
152 dynamicIndex = tonumber(dynamicIndex)
|
Nenue@74
|
153 dynamicSubIndex = tonumber(dynamicSubIndex)
|
Nenue@74
|
154 local cache = kb.DynamicSpells[dynamicType]
|
Nenue@74
|
155 print('type:', dynamicType)
|
Nenue@74
|
156 if dynamicIndex and cache[dynamicIndex] then
|
Nenue@74
|
157 info = kb.DynamicSpells[dynamicType][dynamicIndex]
|
Nenue@74
|
158 print('index:', dynamicIndex)
|
Nenue@74
|
159 if dynamicSubIndex and info[dynamicSubIndex] then
|
Nenue@74
|
160 info = info[dynamicSubIndex]
|
Nenue@74
|
161 print('sub-index:', dynamicSubIndex)
|
Nenue@74
|
162 end
|
Nenue@74
|
163 isAvailable = true
|
Nenue@74
|
164 end
|
Nenue@74
|
165 end
|
Nenue@74
|
166 if not info then
|
Nenue@74
|
167 info = SkeletonKey_GetGenericSpell(spellName, spellID, internalIcon or icon)
|
Nenue@74
|
168 end
|
Nenue@74
|
169 end
|
Nenue@74
|
170 info.isAvailable = isAvailable
|
Nenue@74
|
171
|
Nenue@74
|
172 print('|cFF00FF88Slot Details:|r', info.actionName, info.actionID, info.dynamicType, info.isAvailable)
|
Nenue@74
|
173 for k,v in pairs(info) do
|
Nenue@74
|
174 --cprint(' ',k,v)
|
Nenue@74
|
175 self[k] = v
|
Nenue@74
|
176 end
|
Nenue@74
|
177
|
Nenue@74
|
178 return info
|
Nenue@74
|
179 end
|
Nenue@70
|
180
|
Nenue@70
|
181
|
Nenue@70
|
182 kb.ApplyTalentBinding = function(talentInfo, cache)
|
Nenue@70
|
183 talentInfo.assignedKeys = talentInfo.assignedKeys or {}
|
Nenue@70
|
184 for i , key in pairs(talentInfo.assignedKeys) do
|
Nenue@70
|
185 local command = CLICK_KEYBINDER_KEY.. talentInfo.actionName
|
Nenue@70
|
186 SetBinding(key, command)
|
Nenue@70
|
187 cprint(' **', i, '->', command)
|
Nenue@70
|
188 tinsert(cache, talentInfo)
|
Nenue@70
|
189 end
|
Nenue@70
|
190 end
|
Nenue@70
|
191 kb.CacheTalentBinding = function(talentInfo, cache)
|
Nenue@70
|
192
|
Nenue@70
|
193 local spellID = talentInfo.actionID
|
Nenue@70
|
194 cache[spellID] = cache[spellID] or {}
|
Nenue@70
|
195 cache[spellID] = talentInfo
|
Nenue@70
|
196 cprint(spellID, unpack(kb.TalentBindings[spellID]))
|
Nenue@70
|
197 end
|
Nenue@70
|
198
|
Nenue@70
|
199
|
Nenue@70
|
200 do
|
Nenue@76
|
201 local PROFILE_VERSION = 320
|
Nenue@70
|
202 local commandActions = {}
|
Nenue@70
|
203 local bindings = kb.bindings
|
Nenue@70
|
204 local key, macro = SkeletonKeyKey, SkeletonKeyMacro
|
Nenue@70
|
205 kb.LoadBinding = function( configTable)
|
Nenue@70
|
206 if configTable.command then
|
Nenue@70
|
207 configTable.command = configTable.command:gsub('KeyBinder', 'SkeletonKey')
|
Nenue@70
|
208 end
|
Nenue@70
|
209
|
Nenue@70
|
210 local command, name, icon, actionType, actionID, macroName, macroText =
|
Nenue@70
|
211 configTable.command, configTable.actionName, configTable.iconPath, configTable.actionType,
|
Nenue@70
|
212 configTable.actionID, configTable.macroName, configTable.macroText
|
Nenue@70
|
213
|
Nenue@70
|
214
|
Nenue@70
|
215 local indexKey = actionType..'_'..actionID
|
Nenue@70
|
216 local actionPrefix = "*"..actionType.."-"
|
Nenue@70
|
217 local button = SkeletonKeyKey
|
Nenue@72
|
218 local isAvailable
|
Nenue@70
|
219 local specialButtonType
|
Nenue@70
|
220 if actionType == 'spell' then
|
Nenue@72
|
221 cprint(GetSpellInfo(actionID))
|
Nenue@71
|
222 local dynamicInfo = kb.DynamicSpells[name]
|
Nenue@70
|
223 if dynamicInfo then
|
Nenue@76
|
224 configTable.assignedKeys = configTable.assignedKeys or {GetBindingKey(configTable.command) }
|
Nenue@76
|
225
|
Nenue@72
|
226 cprint('|cFF00FFFFDynamicInfo:|r', dynamicInfo.dynamicType, table.concat(configTable.assignedKeys, ','))
|
Nenue@70
|
227 for k, v in pairs(dynamicInfo) do
|
Nenue@71
|
228 --cprint(' --', k, v)
|
Nenue@70
|
229 configTable[k] = v
|
Nenue@70
|
230 end
|
Nenue@76
|
231 isAvailable = configTable.isAvailable
|
Nenue@76
|
232 elseif GetSpellInfo(actionID) then
|
Nenue@76
|
233 isAvailable = true
|
Nenue@76
|
234
|
Nenue@70
|
235 end
|
Nenue@75
|
236 elseif actionType == 'item' then
|
Nenue@75
|
237 actionID = configTable.actionName
|
Nenue@75
|
238 isAvailable = true
|
Nenue@75
|
239 else
|
Nenue@71
|
240
|
Nenue@70
|
241 if actionType ~= 'macro' then
|
Nenue@70
|
242 actionPrefix = '*macrotext-'
|
Nenue@70
|
243 end
|
Nenue@70
|
244
|
Nenue@70
|
245 specialButtonType = 'macro'
|
Nenue@73
|
246 isAvailable = true
|
Nenue@70
|
247 end
|
Nenue@70
|
248
|
Nenue@72
|
249 if isAvailable then
|
Nenue@70
|
250
|
Nenue@72
|
251 local attributeSuffix, attributeValue, command, target, button = kb.RegisterAction(actionType, actionID, name)
|
Nenue@72
|
252 local actionKey = actionPrefix .. attributeSuffix
|
Nenue@72
|
253 cprint('|cFF00FF88LoadBinding()|r', button:GetName(), "*type-"..attributeSuffix, actionType, '|cFFFFFF00'..actionKey, attributeValue, isAvailable)
|
Nenue@70
|
254
|
Nenue@70
|
255
|
Nenue@70
|
256 kb.SecureAttribute(button, "*type-"..attributeSuffix, specialButtonType or actionType)
|
Nenue@70
|
257 kb.SecureAttribute(button, actionKey, attributeValue)
|
Nenue@72
|
258
|
Nenue@72
|
259 cprint('|cFFFF4400add', name, isAvailable, indexKey, unpack(configTable.assignedKeys))
|
Nenue@70
|
260 kb.bindings[indexKey] = configTable.assignedKeys
|
Nenue@70
|
261 commandActions[command] = kb.bindings[indexKey]
|
Nenue@70
|
262 return command, kb.bindings[indexKey]
|
Nenue@70
|
263 else
|
Nenue@72
|
264 if kb.bindings[indexKey] then
|
Nenue@72
|
265 cprint('|cFFFF4400remove', name, isAvailable, indexKey, unpack(configTable.assignedKeys))
|
Nenue@72
|
266 kb.bindings[indexKey] = nil
|
Nenue@72
|
267 end
|
Nenue@72
|
268
|
Nenue@70
|
269 return nil
|
Nenue@70
|
270 end
|
Nenue@70
|
271 end
|
Nenue@70
|
272
|
Nenue@71
|
273
|
Nenue@70
|
274 local usedSlots = {}
|
Nenue@71
|
275 kb.UpgradeProfile = function(profile)
|
Nenue@70
|
276 wipe(usedSlots)
|
Nenue@70
|
277 for slot, configTable in pairs(profile.buttons) do
|
Nenue@70
|
278
|
Nenue@71
|
279 -- convert old style table
|
Nenue@70
|
280 if #configTable >= 1 then
|
Nenue@70
|
281 local command, name, icon, actionType, actionID, macroName, macroText, spellbookSlot, spellbookType = unpack(configTable)
|
Nenue@70
|
282
|
Nenue@70
|
283 cprint('|CFFFF4400Fixing pad entry', slot, command, name)
|
Nenue@70
|
284 local assignedKeys = {GetBindingKey(command)}
|
Nenue@70
|
285 for k,v in pairs(profile.bindings) do
|
Nenue@70
|
286 if v == command then
|
Nenue@70
|
287 tinsert(assignedKeys, k)
|
Nenue@70
|
288 end
|
Nenue@70
|
289 end
|
Nenue@70
|
290
|
Nenue@70
|
291 configTable = {
|
Nenue@70
|
292 command = command,
|
Nenue@70
|
293 actionType = actionType,
|
Nenue@70
|
294 actionName = name,
|
Nenue@70
|
295 actionID = actionID,
|
Nenue@70
|
296 macroName = macroName,
|
Nenue@70
|
297 macroText = macroText,
|
Nenue@70
|
298 iconPath = icon,
|
Nenue@70
|
299 spellbookSlot = spellbookSlot,
|
Nenue@70
|
300 spellbookType = spellbookType,
|
Nenue@70
|
301 assignedKeys = assignedKeys
|
Nenue@70
|
302 }
|
Nenue@71
|
303
|
Nenue@71
|
304 local dynamic = kb.DynamicSpells[name]
|
Nenue@71
|
305 if dynamic then
|
Nenue@71
|
306 configTable.dynamicType = dynamic.dynamicType
|
Nenue@71
|
307 configTable.dynamicIndex = dynamic.dynamicIndex
|
Nenue@71
|
308 configTable.dynamicSubIndex = dynamic.dynamicSubIndex
|
Nenue@71
|
309 configTable.dynamicID = dynamic.dynamicID
|
Nenue@71
|
310 if configTable.dynamicType == 'talent' then
|
Nenue@71
|
311 profile.talents[name] = configTable
|
Nenue@71
|
312 end
|
Nenue@71
|
313 end
|
Nenue@71
|
314
|
Nenue@71
|
315
|
Nenue@70
|
316 kb.currentProfile.buttons[slot] = configTable
|
Nenue@70
|
317 end
|
Nenue@70
|
318 if not configTable.actionID then
|
Nenue@70
|
319 configTable.actionID = configTable.actionName
|
Nenue@70
|
320 end
|
Nenue@70
|
321 if not configTable.iconPath then
|
Nenue@70
|
322 print('corrected missing icon')
|
Nenue@70
|
323 configTable.iconPath = GetSpellTexture(configTable.actionName)
|
Nenue@70
|
324 end
|
Nenue@70
|
325
|
Nenue@70
|
326 usedSlots[configTable.actionName or configTable.actionID] = true
|
Nenue@70
|
327 usedSlots[configTable.command] = true
|
Nenue@70
|
328 end
|
Nenue@70
|
329
|
Nenue@70
|
330
|
Nenue@71
|
331 -- clean up legacy data
|
Nenue@70
|
332 for spellName, talentInfo in pairs(profile.talents) do
|
Nenue@70
|
333 if not usedSlots[spellName] then
|
Nenue@70
|
334 cprint('|cFFFF4400Unslotted talent', spellName)
|
Nenue@70
|
335 profile.talents[spellName] = nil
|
Nenue@70
|
336 end
|
Nenue@70
|
337 end
|
Nenue@70
|
338 for command in pairs(profile.bound) do
|
Nenue@70
|
339 if not usedSlots[command] then
|
Nenue@70
|
340 cprint('|cFFFF4400Unslotted bound entry', command)
|
Nenue@70
|
341 profile.bound[command] = nil
|
Nenue@70
|
342 profile.commands[command] = nil
|
Nenue@70
|
343 end
|
Nenue@70
|
344 end
|
Nenue@70
|
345 for command in pairs(profile.commands) do
|
Nenue@70
|
346 if not usedSlots[command] then
|
Nenue@70
|
347 cprint('|cFFFF4400Unslotted command entry', command)
|
Nenue@70
|
348 profile.commands[command] = nil
|
Nenue@70
|
349 end
|
Nenue@70
|
350 end
|
Nenue@76
|
351
|
Nenue@76
|
352 if profile.talents then
|
Nenue@76
|
353 profile.talents = nil
|
Nenue@76
|
354 end
|
Nenue@76
|
355
|
Nenue@76
|
356
|
Nenue@76
|
357 profile.versionID = PROFILE_VERSION
|
Nenue@71
|
358 end
|
Nenue@70
|
359
|
Nenue@71
|
360 kb.ApplyBindings = function (profile)
|
Nenue@71
|
361 cprint('|cFF0088FFApplyBindings()')
|
Nenue@71
|
362 --cprint('binding profile', profile)
|
Nenue@71
|
363
|
Nenue@71
|
364 local version = profile.versionID or 0
|
Nenue@76
|
365 if version < PROFILE_VERSION then
|
Nenue@71
|
366 kb.UpgradeProfile(profile)
|
Nenue@71
|
367 end
|
Nenue@71
|
368
|
Nenue@71
|
369
|
Nenue@71
|
370 -- then buttons
|
Nenue@71
|
371 for slot, configTable in pairs(profile.buttons) do
|
Nenue@71
|
372 -- convert old style table
|
Nenue@71
|
373 if kb.LoadBinding(configTable) then
|
Nenue@76
|
374
|
Nenue@71
|
375 if not configTable.assignedKeys then
|
Nenue@75
|
376 configTable.assignedKeys = {GetBindingKey(configTable.command)}
|
Nenue@71
|
377 end
|
Nenue@75
|
378 --if configTable.dynamicType == 'talent' then
|
Nenue@75
|
379 -- kb:print(table.concat(configTable.assignedKeys, ', ') .. ' bound to '.. configTable.actionName)
|
Nenue@75
|
380 --end
|
Nenue@74
|
381 for _, key in pairs(configTable.assignedKeys) do
|
Nenue@76
|
382 local command = configTable.command
|
Nenue@76
|
383 cprint('|cFF00FFFF'.. key .. '|r to|cFF00FF00', command)
|
Nenue@76
|
384 SetBinding(key, command)
|
Nenue@76
|
385 if commandActions[command] and not tContains(commandActions[command], key) then
|
Nenue@76
|
386 tinsert(commandActions[command], key)
|
Nenue@76
|
387 end
|
Nenue@74
|
388 end
|
Nenue@74
|
389
|
Nenue@71
|
390 end
|
Nenue@71
|
391 end
|
Nenue@70
|
392 end
|
Nenue@70
|
393
|
Nenue@70
|
394 kb.ApplyAllBindings =function ()
|
Nenue@74
|
395 print('|cFFFFFF00ApplyAllBindings()')
|
Nenue@70
|
396 wipe(kb.TalentBindings)
|
Nenue@70
|
397 wipe(kb.bindings)
|
Nenue@70
|
398 --kb:print('Loading binding profile', kb.profileName)
|
Nenue@70
|
399
|
Nenue@70
|
400 -- reflect action key settings
|
Nenue@70
|
401 if GetCVarBool("ActionButtonUseKeyDown") then
|
Nenue@70
|
402 SkeletonKeyMacro:RegisterForClicks("AnyDown")
|
Nenue@70
|
403 SkeletonKeyKey:RegisterForClicks("AnyDown")
|
Nenue@70
|
404 else
|
Nenue@70
|
405 SkeletonKeyMacro:RegisterForClicks("AnyUp")
|
Nenue@70
|
406 SkeletonKeyKey:RegisterForClicks("AnyUp")
|
Nenue@70
|
407 end
|
Nenue@70
|
408
|
Nenue@70
|
409 for i, profile in ipairs(kb.orderedProfiles) do
|
Nenue@70
|
410 kb.ApplyBindings(profile)
|
Nenue@70
|
411 end
|
Nenue@70
|
412 -- do this after to ensure that profession binds are properly overridden
|
Nenue@70
|
413 kb.UpdateProfessionInfo()
|
Nenue@70
|
414
|
Nenue@70
|
415 SaveBindings(GetCurrentBindingSet())
|
Nenue@72
|
416
|
Nenue@75
|
417
|
Nenue@70
|
418 end
|
Nenue@70
|
419 end
|
Nenue@70
|
420
|
Nenue@70
|
421
|
Nenue@70
|
422 kb.specInfo = {}
|
Nenue@70
|
423 kb.UpdateSpecInfo = function()
|
Nenue@70
|
424 kb.specInfo.id = GetSpecialization()
|
Nenue@70
|
425 kb.specInfo.globalID, kb.specInfo.name, kb.specInfo.desc, kb.specInfo.texture = GetSpecializationInfo(kb.specInfo.id)
|
Nenue@70
|
426 end
|
Nenue@70
|
427
|
Nenue@70
|
428 kb.UpdateMacroInfo = function()
|
Nenue@70
|
429 print('|cFFFFFF00kb.UpdateMacroInfo()|r')
|
Nenue@70
|
430 for index = 1, GetNumMacros() do
|
Nenue@70
|
431 local name = GetMacroInfo(index)
|
Nenue@70
|
432 kb.SecureAttribute(SkeletonKeyMacro, "*type-macro_"..tostring(name), 'macro')
|
Nenue@70
|
433 kb.SecureAttribute(SkeletonKeyMacro, "*macro-macro_"..tostring(name), index)
|
Nenue@70
|
434 end
|
Nenue@70
|
435 end
|
Nenue@70
|
436
|
Nenue@70
|
437 kb.UpdateTalentInfo = function()
|
Nenue@70
|
438 print('|cFFFFFF00kb.UpdateTalentInfo()|r')
|
Nenue@70
|
439 if kb.talentsPushed then
|
Nenue@70
|
440 return
|
Nenue@70
|
441 end
|
Nenue@70
|
442 for row =1, MAX_TALENT_TIERS do
|
Nenue@70
|
443 for col = 1, NUM_TALENT_COLUMNS do
|
Nenue@70
|
444 local talentID, talentName, icon, selected, available, spellID = GetTalentInfo(row, col, 1)
|
Nenue@70
|
445 local talentInfo = kb.TalentCache[spellID] or {}
|
Nenue@70
|
446 if spellID then
|
Nenue@70
|
447 talentInfo.actionType = 'spell'
|
Nenue@70
|
448 talentInfo.actionName = talentName
|
Nenue@70
|
449 talentInfo.dynamicType = 'talent'
|
Nenue@70
|
450 talentInfo.dynamicID = talentID
|
Nenue@70
|
451 talentInfo.dynamicIndex = row
|
Nenue@70
|
452 talentInfo.dynamicSubIndex = col
|
Nenue@70
|
453 talentInfo.actionID = spellID
|
Nenue@70
|
454 talentInfo.isAvailable = selected
|
Nenue@70
|
455 kb.DynamicSpells[spellID] = talentInfo
|
Nenue@70
|
456 kb.DynamicSpells[talentName] = talentInfo
|
Nenue@70
|
457 end
|
Nenue@70
|
458
|
Nenue@70
|
459 --print('Talent ', row, col, spellID, talentName)
|
Nenue@70
|
460 end
|
Nenue@70
|
461 end
|
Nenue@70
|
462
|
Nenue@70
|
463 for row = 1, MAX_PVP_TALENT_TIERS do
|
Nenue@70
|
464 for col = 1, MAX_PVP_TALENT_COLUMNS do
|
Nenue@70
|
465 local id, name, icon, selected, available, spellID, unlocked = GetPvpTalentInfo(row, col, 1)
|
Nenue@70
|
466 if spellID then
|
Nenue@76
|
467 local talentInfo = kb.TalentCache[spellID] or {}
|
Nenue@70
|
468 talentInfo.actionType = 'spell'
|
Nenue@70
|
469 talentInfo.actionName = name
|
Nenue@70
|
470 talentInfo.dynamicType = 'talent'
|
Nenue@70
|
471 talentInfo.dynamicID = id
|
Nenue@76
|
472 talentInfo.dynamicIndex = row
|
Nenue@76
|
473 talentInfo.dynamicSubIndex = col
|
Nenue@70
|
474 talentInfo.actionID = spellID
|
Nenue@70
|
475 talentInfo.isAvailable = selected
|
Nenue@70
|
476 kb.DynamicSpells[spellID] = talentInfo
|
Nenue@70
|
477 kb.DynamicSpells[name] = talentInfo
|
Nenue@70
|
478 end
|
Nenue@70
|
479 end
|
Nenue@70
|
480 end
|
Nenue@70
|
481
|
Nenue@70
|
482 kb.talentsPushed = true
|
Nenue@70
|
483 kb.UpdateDynamicButtons('talent')
|
Nenue@70
|
484 end
|
Nenue@70
|
485
|
Nenue@70
|
486 kb.UpdateProfessionInfo = function()
|
Nenue@70
|
487 wipe(kb.ProfessionCache)
|
Nenue@70
|
488 local profs = {GetProfessions() }
|
Nenue@70
|
489 --print(GetProfessions())
|
Nenue@70
|
490 local primaryNum = 0
|
Nenue@70
|
491 for i = 1, 6 do
|
Nenue@70
|
492 if profs[i] then
|
Nenue@70
|
493 local profID = profs[i]
|
Nenue@70
|
494 local profName, texture, _, _, numSpells, spellOffset = GetProfessionInfo(profID)
|
Nenue@70
|
495 cprint(i, profID, profName, numSpells, spellOffset)
|
Nenue@70
|
496 if not SECONDARY_PROFESSIONS[profID] then
|
Nenue@70
|
497 primaryNum = primaryNum + 1
|
Nenue@70
|
498 end
|
Nenue@70
|
499 local profNum = SECONDARY_PROFESSIONS[profID] or primaryNum
|
Nenue@70
|
500 cprint(i, profNum)
|
Nenue@70
|
501
|
Nenue@70
|
502
|
Nenue@70
|
503 kb.ProfessionCache[profNum] = kb.ProfessionCache[profNum] or {}
|
Nenue@70
|
504
|
Nenue@70
|
505 for j = 1, numSpells do
|
Nenue@70
|
506 local spellName, _, icon, _, _, _, spellID = GetSpellInfo(spellOffset+j, BOOKTYPE_PROFESSION)
|
Nenue@70
|
507 cprint(j, spellName)
|
Nenue@70
|
508 local profInfo = {
|
Nenue@70
|
509 actionType = 'spell',
|
Nenue@70
|
510 actionName = spellName,
|
Nenue@70
|
511 statusText = 'Profession ' .. i,
|
Nenue@70
|
512 actionID = spellID,
|
Nenue@70
|
513 iconPath = icon,
|
Nenue@70
|
514 dynamicIndex = i,
|
Nenue@70
|
515 dynamicSubIndex = j,
|
Nenue@70
|
516 dynamicType = 'profession',
|
Nenue@70
|
517 spellbookOffset = (spellOffset+j),
|
Nenue@70
|
518 spellbookType = BOOKTYPE_PROFESSION,
|
Nenue@70
|
519 isAvailable = true,
|
Nenue@70
|
520 -- need to check if necessary
|
Nenue@70
|
521 uniqueID = profID,
|
Nenue@70
|
522 }
|
Nenue@70
|
523
|
Nenue@70
|
524 kb.SecureAttribute(SkeletonKeyKey, "*type-profession_"..i .. '_' ..j, "spell")
|
Nenue@70
|
525 kb.SecureAttribute(SkeletonKeyKey, "*spell-profession_"..i .. '_' ..j, spellName)
|
Nenue@70
|
526
|
Nenue@70
|
527 kb.ProfessionCache[spellName] = profInfo
|
Nenue@70
|
528 kb.ProfessionCache[spellID] = profInfo
|
Nenue@70
|
529
|
Nenue@70
|
530 kb.DynamicSpells[spellName] = profInfo
|
Nenue@70
|
531 kb.DynamicSpells[spellID] = profInfo
|
Nenue@70
|
532
|
Nenue@70
|
533 kb.DynamicSpells.profession[i] = kb.DynamicSpells.profession[i] or {}
|
Nenue@70
|
534 kb.DynamicSpells.profession[i][j] = profInfo
|
Nenue@70
|
535 --print(' |cFF0088FF['..i..']|r|cFFFF44BB['..spellOffset+i..']|r', spellName, "profession_"..i .. '_' ..j)
|
Nenue@70
|
536 end
|
Nenue@70
|
537 end
|
Nenue@70
|
538
|
Nenue@70
|
539 end
|
Nenue@70
|
540
|
Nenue@70
|
541 kb.UpdateDynamicButtons('profession')
|
Nenue@70
|
542 end
|
Nenue@70
|
543
|
Nenue@70
|
544
|
Nenue@70
|
545
|
Nenue@70
|
546 kb.UpdatePetInfo = function()
|
Nenue@70
|
547 local hasPetSpells, petType = HasPetSpells()
|
Nenue@70
|
548
|
Nenue@70
|
549 -- reconcile saved data if it becomes available
|
Nenue@70
|
550 if kb.db then
|
Nenue@70
|
551 kb.db.petSpellsDB = kb.db.petSpellsDB or {}
|
Nenue@70
|
552 kb.db.petSpellsDB.subtext = kb.db.petSpellsDB.subtext or {}
|
Nenue@70
|
553 kb.db.petSpellsDB.spell = kb.db.petSpellsDB.spell or {}
|
Nenue@70
|
554 local spellCache = kb.db.petSpellsDB.spell
|
Nenue@70
|
555 local subtextCache = kb.db.petSpellsDB.subtext
|
Nenue@70
|
556 if petSpellCache then
|
Nenue@70
|
557 for k,v in pairs(petSpellCache) do
|
Nenue@70
|
558 if not spellCache[k] then
|
Nenue@70
|
559 spellCache[k] = v
|
Nenue@70
|
560 end
|
Nenue@70
|
561 end
|
Nenue@70
|
562 end
|
Nenue@70
|
563 petSpellCache = spellCache
|
Nenue@70
|
564 if petSubtextCache then
|
Nenue@70
|
565 for k,v in pairs(petSubtextCache) do
|
Nenue@70
|
566 if not subtextCache[k] then
|
Nenue@70
|
567 subtextCache[k] = v
|
Nenue@70
|
568 end
|
Nenue@70
|
569 end
|
Nenue@70
|
570 end
|
Nenue@70
|
571 petSubtextCache = subtextCache
|
Nenue@70
|
572 else
|
Nenue@70
|
573 petSpellCache = {}
|
Nenue@70
|
574 petSubtextCache = {}
|
Nenue@70
|
575 end
|
Nenue@70
|
576
|
Nenue@70
|
577 if PetHasSpellbook() then
|
Nenue@70
|
578 --print('PET SPELLBOOK')
|
Nenue@70
|
579 local spellbookOffset = 1
|
Nenue@70
|
580 local specialNum = {}
|
Nenue@70
|
581 local newSubtextItems = false
|
Nenue@70
|
582
|
Nenue@70
|
583 repeat
|
Nenue@70
|
584
|
Nenue@70
|
585 local spellType, spellID = GetSpellBookItemInfo(spellbookOffset, BOOKTYPE_PET)
|
Nenue@70
|
586 local spellName, subText = GetSpellBookItemName(spellbookOffset, BOOKTYPE_PET)
|
Nenue@70
|
587 local texture = GetSpellBookItemTexture(spellbookOffset, BOOKTYPE_PET)
|
Nenue@70
|
588 if (spellType == 'SPELL') and (not IsPassiveSpell(spellbookOffset, BOOKTYPE_PET)) then
|
Nenue@70
|
589 local info = kb.PetCache[spellName] or {}
|
Nenue@70
|
590 kb.PetCache.spellslot[spellName] = {spellbookOffset, spellName, subText, spellID, texture}
|
Nenue@70
|
591 --print('|cFF00FF88spellslot['..spellName..']|r', '=>', i, subText)
|
Nenue@70
|
592
|
Nenue@70
|
593 if subText then
|
Nenue@70
|
594 kb.PetCache.subtext[subText] = kb.PetCache.subtext[subText] or {}
|
Nenue@70
|
595 specialNum[subText] = (specialNum[subText] or 0) + 1
|
Nenue@70
|
596
|
Nenue@70
|
597 petSpellCache[spellName] = subText
|
Nenue@70
|
598 petSubtextCache[subText] = petSubtextCache[subText] or {}
|
Nenue@70
|
599
|
Nenue@70
|
600 -- add to the list
|
Nenue@70
|
601 if not petSubtextCache[subText][spellName] then
|
Nenue@70
|
602 petSubtextCache[subText][spellName] = true
|
Nenue@70
|
603 newSubtextItems = true
|
Nenue@70
|
604 --print('|cFF00FFFFspecial['..spellName..']|r', '\n','|cFF00FFFFsubtext['..subText..']['..specialNum[subText]..']|r', '=>', i, spellName, subText, spellID, texture, specialNum[subText])
|
Nenue@70
|
605 end
|
Nenue@70
|
606
|
Nenue@70
|
607
|
Nenue@70
|
608
|
Nenue@70
|
609 local entry = {spellbookOffset, spellName, subText, spellID, texture, specialNum[subText] }
|
Nenue@70
|
610
|
Nenue@70
|
611
|
Nenue@70
|
612 info.spellbookOffset = spellbookOffset
|
Nenue@70
|
613 info.spellbookType = BOOKTYPE_PET
|
Nenue@70
|
614 info.actionName = spellName
|
Nenue@70
|
615 info.spellID = spellID
|
Nenue@70
|
616 info.dynamicType = 'petaction'
|
Nenue@70
|
617 info.dynamicID = spellID
|
Nenue@70
|
618 info.dynamicIndex = subText
|
Nenue@70
|
619 info.dynamicSubIndex = specialNum[subText]
|
Nenue@70
|
620 info.isAvailable = true
|
Nenue@70
|
621
|
Nenue@70
|
622 kb.PetCache.special[spellName] = info
|
Nenue@70
|
623 kb.PetCache.subtext[subText][specialNum[subText]] = info
|
Nenue@70
|
624 kb.DynamicSpells[spellName] = info
|
Nenue@70
|
625 kb.DynamicSpells[spellID] = info
|
Nenue@70
|
626
|
Nenue@70
|
627 end
|
Nenue@70
|
628
|
Nenue@70
|
629 if spellID then
|
Nenue@70
|
630 kb.PetCache.spell[spellbookOffset] = {spellID, spellName, subText}
|
Nenue@70
|
631 --print('|cFF0088FFspell['..i..']|r', '=>', spellID, spellName, subText)
|
Nenue@70
|
632 end
|
Nenue@70
|
633
|
Nenue@70
|
634 kb.PetCache[spellName] = info
|
Nenue@70
|
635 end
|
Nenue@70
|
636
|
Nenue@70
|
637 spellbookOffset = spellbookOffset + 1
|
Nenue@70
|
638 until spellType == nil
|
Nenue@70
|
639
|
Nenue@70
|
640 if newSubtextItems then
|
Nenue@70
|
641
|
Nenue@70
|
642 local macrotext = ""
|
Nenue@70
|
643 for subText, spells in pairs(petSubtextCache) do
|
Nenue@70
|
644 if specialNum[subText] then
|
Nenue@70
|
645 for spellName, enabled in pairs(spells) do
|
Nenue@70
|
646 macrotext = macrotext .. "/cast " .. spellName .. "\n"
|
Nenue@70
|
647 end
|
Nenue@70
|
648 kb.SecureAttribute(SkeletonKeyMacro, "*macrotext-petaction_"..subText.."_"..specialNum[subText], macrotext)
|
Nenue@70
|
649 end
|
Nenue@70
|
650 end
|
Nenue@70
|
651 end
|
Nenue@70
|
652
|
Nenue@70
|
653
|
Nenue@70
|
654 else
|
Nenue@70
|
655 --print('NO PET SPELLBOOK')
|
Nenue@70
|
656 wipe(kb.PetCache.spell)
|
Nenue@70
|
657 wipe(kb.PetCache.spellslot)
|
Nenue@70
|
658 end
|
Nenue@70
|
659
|
Nenue@70
|
660 if PetHasActionBar() then
|
Nenue@70
|
661 --print('PET ACTION BAR')
|
Nenue@70
|
662 for i = 1, 10 do
|
Nenue@70
|
663
|
Nenue@70
|
664
|
Nenue@70
|
665 local name, subtext, texture, isToken, isActive = GetPetActionInfo(i)
|
Nenue@70
|
666 if name then
|
Nenue@70
|
667 kb.PetCache.action[i] = {name, subtext, texture, isToken, isActive }
|
Nenue@70
|
668
|
Nenue@70
|
669
|
Nenue@70
|
670 end
|
Nenue@70
|
671 --print('|cFFFFFF00action['..i..']|r', name, subtext, texture)
|
Nenue@70
|
672 end
|
Nenue@70
|
673 else
|
Nenue@70
|
674 --print('NO PET ACTION BAR')
|
Nenue@70
|
675 wipe(kb.PetCache.action)
|
Nenue@70
|
676 end
|
Nenue@70
|
677
|
Nenue@70
|
678 kb.UpdateDynamicButtons('petaction')
|
Nenue@70
|
679
|
Nenue@70
|
680 end
|
Nenue@70
|
681
|
Nenue@70
|
682 kb.UpdateSystemBinds = function()
|
Nenue@70
|
683 wipe(kb.SystemBindings)
|
Nenue@70
|
684 local n = GetNumBindings()
|
Nenue@70
|
685 for i=1, n do
|
Nenue@70
|
686 local command, key1, key2 = GetBinding(i)
|
Nenue@70
|
687 if not command:match('ACTION.*%d+') then
|
Nenue@70
|
688 if key1 then
|
Nenue@70
|
689 kb.SystemBindings[key1] = command
|
Nenue@70
|
690 end
|
Nenue@70
|
691 if key2 then
|
Nenue@70
|
692 kb.SystemBindings[key2] = command
|
Nenue@70
|
693 end
|
Nenue@70
|
694 else
|
Nenue@70
|
695 --print('ignoring action button binding', command)
|
Nenue@70
|
696 end
|
Nenue@70
|
697 end
|
Nenue@70
|
698 end
|
Nenue@70
|
699
|
Nenue@70
|
700 kb.UpdateDynamicButtons = function(dynamicType)
|
Nenue@70
|
701 for i, button in ipairs(kb.buttons) do
|
Nenue@70
|
702 if button.isDynamic == dynamicType then
|
Nenue@70
|
703 kb.UpdateSlot(button, true)
|
Nenue@70
|
704 end
|
Nenue@70
|
705 end
|
Nenue@70
|
706 end
|
Nenue@70
|
707
|
Nenue@70
|
708 kb.pendingAttributes = {}
|
Nenue@70
|
709 kb.SecureAttribute = function(target, name, value)
|
Nenue@70
|
710 if InCombatLockdown() then
|
Nenue@70
|
711 if #kb.pendingAttributes == 0 then
|
Nenue@70
|
712 kb:print(kb.L('Key bindings will be applied when you exit combat.'))
|
Nenue@70
|
713 end
|
Nenue@70
|
714 tinsert(kb.pendingAttributes, {target, name, value})
|
Nenue@70
|
715 SkeletonKey:RegisterEvent('PLAYER_REGEN_ENABLED')
|
Nenue@70
|
716 else
|
Nenue@70
|
717 --cprint('|cFFFF4444' .. target:GetName()..'|r.|cFFFFFF00'.. tostring(name)..'|r = "'..tostring(value)..'"')
|
Nenue@70
|
718 target:SetAttribute(name, value)
|
Nenue@70
|
719 end
|
Nenue@70
|
720 end
|
Nenue@70
|
721
|
Nenue@70
|
722 kb.PLAYER_REGEN_ENABLED = function()
|
Nenue@70
|
723 if #kb.pendingAttributes >= 1 then
|
Nenue@70
|
724 local args = tremove(kb.pendingAttributes)
|
Nenue@70
|
725 while args do
|
Nenue@70
|
726 local target, name, value = unpack(args)
|
Nenue@70
|
727 --print(target:GetName(), 'attribute', '"'.. tostring(name)..'" = "'..tostring(value)..'"')
|
Nenue@70
|
728 cprint('deferred', target:GetName(), 'attribute', '"'.. tostring(name)..'" = "'..tostring(value)..'"')
|
Nenue@70
|
729 target:SetAttribute(name, value)
|
Nenue@70
|
730 args = tremove(kb.pendingAttributes)
|
Nenue@70
|
731 end
|
Nenue@70
|
732 end
|
Nenue@70
|
733
|
Nenue@70
|
734 if #kb.pendingCalls >= 1 then
|
Nenue@70
|
735
|
Nenue@70
|
736 local func = tremove(kb.pendingCalls)
|
Nenue@70
|
737 while func do
|
Nenue@70
|
738 func()
|
Nenue@70
|
739 end
|
Nenue@70
|
740 end
|
Nenue@70
|
741 end
|
Nenue@70
|
742
|
Nenue@70
|
743 kb.UpdateBindingsCache = function(actionType, actionID, bindings)
|
Nenue@70
|
744 local indexKey = actionType .. '_' .. actionID
|
Nenue@70
|
745 kb.bindings[indexKey] = bindings
|
Nenue@70
|
746
|
Nenue@70
|
747 cprint('|cFF00FF00'..indexKey..'|r = {'.. table.concat(bindings,', ').. '}')
|
Nenue@70
|
748 tinsert(kb.ChangedBindings, {actionType, actionID})
|
Nenue@70
|
749 end |