|This article documents API changes made in patch 2.0.1.
- 1 Interface AddOn Kit
- 2 Significant Changes
- 3 Protected Code
- 4 Lua environment
- 5 UI Performance
- 6 Slash Commands
- 7 Conditional Macro Commands
- 8 Sequenced Casting
- 9 Macro Feedback
- 10 Spell Casting
- 11 Secure Templates
- 12 Actions
- 13 API Changes
- 14 API Changes: Strings
- 15 API Changes: Bindings
- 16 API Changes: Security and Protection
- 17 API Changes: Information
- 18 API Changes: Spell Information
- 19 API Changes: Spell Casting
- 20 API Changes: Auto Looting
- 21 API Changes: Attacking
- 22 API Changes: Active Buffs/Auras
- 23 API Changes: Party Management
- 24 API Changes: Macros
- 25 Frame Features
- 26 Frame Methods
- 27 XML Improvements
- 28 Event Changes
- 29 Protected Frame Features
- 30 Key Bindings
- 31 Saved Variables
- 32 Bug Fixes 2.0.1
- 33 Bug Fixes 2.0.3
- 34 Bug Fixes 2.0.4
- 35 Bug Fixes 2.0.6
This is a consolidated list of the announced (and sometimes observed) changes in the User Interface API's and functionality for the Burning Crusade expansion (2.0) release. Please note that this thread is to discuss the upcoming changes and any clarifications or features that are a direct result of those changes, or things which we've been asked to remind slouken of.
WoW 2.0 represents a major change in the UI code, and as such ALL AddOns will need to be updated before they will work, if for no other reason than the Lua 5.0 to Lua 5.1 changes. Some AddOns will be more seriously affected, and a few will have to be redesigned.
Interface AddOn Kit
The Burning Crusade BETA Interface AddOn Kit can be found here (Mac version)
- WoW 2.0 has upgraded to Lua 5.1.1 as it provides a number of benefits (notably incremental garbage collection and efficient varargs). There ARE some incompatible changes from Lua 5.0, authors should read the Lua manual; main points include:
- Iteration over tables in a for loop now requires use of the pairs function: thus,
for k,v in table dobecomes
for k,v in pairs(table) do
- Size of the array portion of the table is no longer stored in the n key: thus,
size = t.n or table.getn(t)becomes
size = #tusing the new # size operator.
- Functions may now use the vararg argument (...) to take a variable number of arguments without having to declare them in the function signature.
- Iteration over tables in a for loop now requires use of the pairs function: thus,
- There is a new unit "focus" which behaves like target, the "PLAYER_FOCUS_CHANGED" event is fired when it is changed, and you receive unit events for this unit.
- The protected code mechanism introduced in 1.10 has been extended to cover more functions. Blizzard code is considered secure, all other code is untrusted.
The following function types are restricted for AddOns;
- Movement is never allowed.
- Spell casting and targetting (including focus) are only allowed using secure templates or special slash commands (not /script).
- Programatic modification of macros, key bindings, and the action bar page only allowed outside of combat.
- Trading, placing auction bids, querying LFG, crafting, and reloading the UI are only allowed interactively.
- Action buttons and targeting frames are now "protected". Protected frames cannot be programmatically moved, shown or hidden, or have their attributes (unit, action, etc) changed during combat.
- Dueling, dismissing your pet, and abandoning your pet are no longer protected functions. *2.0.3* - Possibly not working right
- Script handlers defined in SecureXML templates remain secure when they are inherited, unless overridden by AddOn code.
- The parent of a protected frame is implicitly protected also, as are any frames which it is anchored to.
- The Lua coroutine library is now available - Coroutines create a new Lua state and should be used sparingly to conserve memory.
Performance of some elements of the UI engine has been improved:
- Frame method invocation is significantly faster (more than twice as fast in some cases!)
- Handler dispatch has been optimized, especially for simple handlers such as OnUpdate.
The following new slash commands will be available:
- Target Cycling:
/targetraid- All work using the TargetNearest* API's and enemy/friend may only be called once per macro. An optional argument of '1' can be used to reverse direction.
- Items and equipment:
/use <bagid> <bagslot>,
/equipslot <slot> <itemname>,
/userandom <item1>, <item2>, <item3>will attempt to use a randomly selected one of the specified items
- Pet control:
/stopcastingwill cancel the spell that is currently being cast,
/castrandom <spell1>, <spell2>, <spell3>will attempt to cast a randomly selected one of the specified spells
- Action bar:
/swapactionbar <num1> <num2>
/cancelaura <name>- cancels an aura as if it were right clicked
/click ButtonName [mousebutton]
Also note the following clarifications and details:
/startattacktake all valid unit ids (as well as entity names in the case of
/target) Where applicable they default to target if no unit is specified.
- In WoW 2.0, feeding your pet, poisons, etc. will be even easier to macro:
/cast Feed Petthen
/use Dry Pork Ribs. You will also be able to click on food in your action bar when you are targeting Feed Pet and other item targeting spells.
- The first cast in a macro that fails will prevent further casts in the macro as if the 1.5 second global cooldown had been triggered.
- Item names can be used interchangably with spell names in
/petattackwill work like normal with no arguments, but if they're given arguments its interpreted and used as a unit ID or name to attack (you will not attack if the target cannot be found). *2.0.3*
- If a unit is passed to
/petattackYOUR target will not be changed. *2.0.3*
Conditional Macro Commands
/click commands can be given multiple actions and conditions on which to use those actions. The first matching action will be used.
/petattack commands also support the
[target=unit] option syntax. *2.0.3*
The syntax is as follows:
/command [option] action; [option1,option2] action
If multiple conditions are provided for an action, they must all be met before the action will happen. The conditions supported are as follows:
- target=unit - Run the command and perform tests on the specified unit (only applicable to spell casting or item use). You can specify 'none' as a target unit to act as if you have no target.
- help/harm - Choose depending on whether target is assistable or attackable
- combat - Choose depending on whether you're in combat or not
- stance or stance:# - Choose depending on your stance/shapeshift ("stance" alone matches any stance)
- stealth - Choose depending on whether you are stealthed
- mounted - Choose depending on whether you are mounted
- swimming - Choose depending on whether you are swimming
- flying - Choose depending on whether you are flying
- indoors - Choose depending on whether you are indoors
- outdoors - Choose depending on whether you are outdoors
- modifier or modifier:shift|ctrl|alt - Choose depending on any modifier keys you're holding at the time the macro runs
- equipped:<invslot>|<itemclass>|<itemsubclass> - Choose if you have the specific item type equipped
- actionbar:<bar> - Choose if a specific actionbar is shown
- button:Z - Choose if the specified button is held down
- pet:<name>|<family> - Choose if your current active pet has the specified name or family (e.g. pet:bear/wolf)
- channeling or channeling:spell - Choose if you are channeling (possibly a specific spell)
- exists - Choose if the target exists
- dead - Choose if the target is dead
For stance and modifier you can list multiple matching values separated by the / character for the condition to be true if ANY of them are true. For example
modifier:shift/ctrl matches if shift or control is held down.
Any condition can be prefixed with 'no' to select if it does NOT match. For example
- There is a new
/stopmacro [condition,...]command that stops macro execution if the specified conditions are met. e.g.
- The pet action commands (
/petattacketc) also allow for conditional execution as above.
- As a special case, if you
target=target, it will use the argument as the target instead. e.g.
/focus [target=focus,noexists] petwill focus on the pet if the focus doesn't exist. *2.0.3*
- For the player, pet, party members and pets, and raid members and pets, you can now substitute the name of the unit anywhere a unit token is normally used. *2.0.3*
- In battlegrounds you can suffix the name with -<realm> (e.g. -Silver Hand) to reference a player from a specific realm. *2.0.3*
- You can suffix name or unit ID with -target to get the unit's target, and (for party and raid members) -pet to get the unit's pet. *2.0.3*
- A new command
/castsequence reset=N/target/combat/shift/alt/ctrl <spell1>, <spell2>, <spell3>
- The reset line can specify a number of seconds after which a sequence resets, or if it should reset on target change or leaving combat.
- The sequence tracks the 'next' spell in the sequence until it resets, the next spell only advances on a successful cast.
- You can specify a conditional at the start of the command before the reset to filter whether the sequence is used (You cannot use per-spell conditionals)
- You can specify items as well as spells
- It will equip items if they need to be equipped, and use them otherwise. *2.0.3*
-are comment prefixes in macros
- A macro that starts with
show <item or spell>shows feedback for that spell,
show noneshows no feedback.
- A macro that starts with
showtooltip <item or spell>shows the tooltip for that spell,
showtooltip noneshows no feedback. *2.0.3*
showtooltipshould come before
- Otherwise the first
/randomusecommand is used to select the feedback spell
- Sequences will cycle the item/spell they show feedback for to match the next in sequence.
- A new question mark macro icon is available, selecting that will cause the feedback spell/item's icon to be used.
- When casting a ranked buff spell that's too high a level for a friendly target the game will automatically use the highest appropriate rank of the spell instead.
- Toggleable abilities will have a 1.5 second delay after being cast before they can be cancelled by clicking on them, to prevent accidentally canceling them. You can still right click a buff to cancel it anytime.
There are several secure templates that AddOns can use to create new functional buttons. Many of them have extensive documentation in the FrameXML .lua files.:
- The secure button templates have support for pre-loaded items which are used as spell targets for feeding, poisoning, etc.
- Added an action type "attribute" to secure action buttons which sets an arbitrary attribute on an arbitrary frame. *2.0.3*
- Added support for state header resizing and re-parenting on state changes. *2.0.3*
- The old
CURRENT_ACTIONBAR_PAGEvariable is no longer used. There's a new slash command
/changeactionbar <num>. Macros that used to do
/script CURRENT_ACTIONBAR_PAGE=2 ChangeActionBarChange()can now simply do
- Action buttons no longer use shift-click to pick up actions, but instead use shift-drag. This frees up the shift modifier for use in macros or as a self-cast modifier. *2.0.3*
- You no longer need to strip special codes from item links before passing them to item information functions.
- Item information functions accept an item ID, an item link, or the name of an item in your inventory.
- Inventory functions will accept a slot name in addition to a slot number.
- Spell information functions take a spell name, or the spell ID followed by "pet" or "spell"; e.g.
IsSpellInRange(2, "pet"); (Updated 10/13)
API Changes: Strings
... = strsplit("delimiters", "string" [,"limit"])-- Split a string on any of the delimiter characters.
string = strjoin("separator", ...)-- Join strings with a separator
string = strconcat(...)-- Join strings without a separator
string = strtrim("string"[,"characters"])-- Trim characters from the ends of a string (defaults to whitespace charaters)
string.joinare also references to the new functions.
API Changes: Bindings
- NEW SetBindingSpell("KEY", "Spell Name") -- Set a key binding directly to a spell, uses the same spell name syntax as
- NEW SetBindingClick("KEY", "ButtonName" [,"mouseButton"]) -- Set a key binding directly to a Button object. The click sends a mouse down when the key is pressed, and a mouse up when it is released.
- NEW SetBindingItem("KEY", "itemname")
- NEW SetBindingMacro("KEY", "macroname"|macroid)
- NEW SetOverrideBinding(owner, isPriority, "KEY" [,"COMMAND"]) -- Set (or clear) an override key binding.
- NEW SetOverrideBindingSpell(owner, isPriority, "KEY", "spellname")
- NEW SetOverrideBindingClick(owner, isPriority, "KEY", "ButtoName"[,"mouseButton"])
- NEW SetOverrideBindingItem(owner, isPriority, "KEY", "itemname")
- NEW SetOverrideBindingMacro(owner, isPriority, "KEY", "macroname"|macroid)
- NEW ClearOverrideBindings(owner) -- Reset all overrides belonging to an owner.
- NEW SetMouselookOverrideBinding("KEY" [,"COMMAND"])
- UPDATED action = GetBindingAction("KEY" [,checkOverride])
- NEW IsModifierKeyDown() -- equivalent to (IsShiftKeyDown() or IsControlKeyDown() or IsAltKeyDown())
- NEW IsMouseButtonDown([button or "button"])
- NEW button = GetMouseButtonClicked() -- Returns the name of the button that triggered a mouse down/up/click/doubleclick event. *2.0.3*
- NEW hasMacro = CursorHasMacro() *2.0.3*
- NEW type, detail[,subdetail] = GetCursorInfo() -- one of "item",id,link / "spell",slot,"spell"|"pet" / "macro",index / "money",amount *2.0.3*
API Changes: Security and Protection
- NEW isSecure = issecure() -- Determine if the current environment is secure
- NEW isSecure = issecurevariable([table,] "name") -- Determine if the specified variable is secure.
- NEW ... = securecall(function or "functionName", ...) -- Call a function from a secure environment without risking secure status.
- NEW hooksecurefunc([table,] "functionName", hookfunc) -- Creates a secure 'post hook' for the named function. The hookfunc is invoked after the original function, and receives the same parameters. Return values from hookfunc are discarded. This is the only safe way to hook functions that execute protected functionality.
- NEW locked = InCombatLockdown() -- Returns true if the in-combat AddOn restrictions are active.
API Changes: Information
- NEW GetCritChance()
- NEW GetRangedCritChance()
- NEW GetSpellCritChance(school)
- UPDATED itemName, itemLink, itemRarity, itemLevel, itemMinLevel, itemType, itemSubType, itemStackCount, itemEquipLoc, invTexture = GetItemInfo(itemID|"itemLink") -- Now returns item level WARNING - This change is not backward compatible
- NEW count = GetItemCount(itemID|"itemLink")
- NEW spell, rank, displayName, icon, startTime, endTime, isTradeSkill = UnitCastingInfo("unit")
- NEW spell, rank, displayName, icon, startTime, endTime, isTradeSkill = UnitChannelInfo("unit")
- NEW type, id, subtype = GetActionInfo(slot)
- NEW startTime, duration, enable = GetItemCooldown(item)
- NEW usable, noMana = IsUsableItem(item)
- NEW consumable = IsConsumableItem(item)
- NEW equipped = IsEquippedItem(item)
- NEW casting = IsCurrentItem(item)
- NEW hasRange = ItemHasRange(item)
- NEW result = IsItemInRange(item, [unit]) - nil for no valid target, 0 for out of range, 1 for in range
- UPDATED result = IsActionInRange(slot,[unit])
- NEW isEquipped = IsEquippedItemType("type") - where "type" is any valid inventory type, item class, or item subclass
- NEW swimming = IsSwimming()
- NEW flying = IsFlying()
- NEW mounted = IsMounted()
- NEW afk = UnitIsAFK("unit") -- Only works for friendly units
- NEW dnd = UnitIsDND("unit")
API Changes: Spell Information
- RENAMED isPassive = IsPassiveSpell(spell) -- Formerly IsSpellPassive(spell)
- NEW isAttack = IsAttackSpell(spell)
- NEW isAutoRepeating = IsAutoRepeatSpell(spell)
- NEW usable, noMana = IsUsableSpell(spell)
- NEW hasRange = SpellHasRange(spell)
- NEW result = IsSpellInRange(spell, [unit]) - is nil for no valid target, 0 for out of range, 1 for in range
- NEW name, rank = GetItemSpell(item)
- NEW index, item, spell = QueryCastSequence(sequence)
API Changes: Spell Casting
- NEW SetActionSelfCastKey("SHIFT" or "CTRL" or "ALT")
- NEW key = GetActionSelfCastKey()
- NEW down = IsActionSelfCastKeyDown()
API Changes: Auto Looting
- NEW SetAutoLootDefault(enabled)
- NEW enabled = GetAutoLootDefault()
- NEW SetAutoLootToggleKey("SHIFT" or "CTRL" or "ALT")
- NEW key = GetAutoLootToggleKey()
- NEW down = IsAutoLootKeyDown()
API Changes: Attacking
- NEW StopAttack()
API Changes: Active Buffs/Auras
- NEW name, rank = GetPlayerBuffName(id or "name"[,"rank"]);
- UPDATED texture = GetPlayerBuffTexture(id or "name"[,"rank"]) - Added name/rank
- UPDATED type = GetPlayerBuffDispelType(id or "name"[,"rank"]) - as above
- UPDATED count = GetPlayerBuffApplications(id or "name"[,"rank"]) - as above
- UPDATED CancelPlayerBuff(id or "name"[,"rank"]) - as above
- UPDATED name,rank,icon,count = UnitBuff("unit", id or "name"[,"rank"]) - as above
- UPDATED name,rank,icon,count,type = UnitDebuff("unit", id or "name"[,"rank"]) - as above
API Changes: Party Management
- NEW InviteUnit("unit" or "name") - Replaces InviteByName/InviteToParty
- NEW UninviteUnit("unit" or "name") - Replaces UninviteByName/UninviteFromParty
API Changes: Macros
- UPDATED name,texture,body,isLocal = GetMacroInfo(id|"name") - Now takes ID or name
- UPDATED DeleteMacro() -- as above
- UPDATED EditMacro() -- as above
- UPDATED PickupMacro() -- as above
- Frames can now specify multiple comma separated templates in the inherits XML attribute, and will inherit each of them in the order they appear
- Frame handlers now are passed parameters in addition to setting the global variables when executed. Accessing these parameters locally is faster than accessing the globals, and is recommended.
- Frames no longer share a single virtual metatable, instead each type has its own metatable with a pre-populated index for faster method lookups.
- In addition to OnClick, Button frames also have PreClick and PostClick handlers, invoked before and after the OnClick respectively.
- GameTooltips now automatically create new lines as needed.
- Script handlers defined in XML are implicitly wrapped with a function declaration (and trailing end) as follows (NOTE: AddOns that hook script handlers must now pass these parameters):
- default: function(self)
- OnAttributeChanged: function(self,name,value)
- OnChar: function(self,text)
- OnCharComposition: function(self,text)
- OnClick: function(self,button,down)
- OnColorSelect: function(self,r,g,b)
- OnCursorChanged: function(self,x,y,w,h)
- OnDoubleClick: function(self,button)
- OnDragStart: function(self,button)
- OnEnter: function(self, motion)
- OnEvent: function(self,event,...)
- OnHorizontalScroll: function(self,offset)
- OnHyperlinkClick: function(self,link,text,button)
- OnHyperlinkEnter: function(self,link,text)
- OnHyperlinkLeave: function(self,link,text)
- OnInputLanguageChanged: function(self,language)
- OnKeyDown: function(self,key)
- OnKeyUp: function(self,key)
- OnLeave: function(self, motion)
- OnMouseDown: function(self,button)
- OnMouseUp: function(self,button)
- OnMouseWheel: function(self,delta)
- OnMovieShowSubtitle: function(self,text)
- OnScrollRangeChanged: function(self,xrange,yrange)
- OnSizeChanged: function(self,w,h)
- OnTooltipAddMoney: function(self,cost)
- OnUpdate: function(self,elapsed)
- OnValueChanged: function(self,value)
- OnVerticalScroll: function(self,offset)
- PostClick: function(self,button,down)
- PreClick: function(self,button,down)
- GameTooltip objects have the following new script handlers
- NEW OnTooltipSetUnit - Triggered when a Tooltip shows a unit *2.0.3*
- NEW OnTooltipSetItem - Triggered when a Tooltip shows an item *2.0.3*
- NEW OnTooltipSetSpell - Triggered when a Tooltip shows a spell *2.0.3*
- NEW minx,miny,width,height = Region:GetRect()
- NEW minx,miny,width,height = Frame:GetBoundsRect()
- NEW isProtected = Frame:IsProtected()
- NEW canChange = Frame:CanChangeProtectedState()
- NEW Frame:SetAttribute("name", value) -- Set an attribute on a frame, for protected frames this method is protected in combat.
- NEW value = Frame:GetAttribute("name")
- NEW Frame:AllowAttributeChanges() -- Temporarily allow attribute changes during combat, this function is protected.
- UPDATED Button:RegisterForClicks("button"[,"button"...]) -- Now accepts "AnyUp" and "AnyDown" to receive all buttons' up and downs.
- You can set initial attributes on frames from frames from XML: <Attributes><Attribute name="foo" type="nil|boolean|number|string" value="bar"/></Attributes>
There are a number of new events:
- UNIT_TARGET, "unit" is fired when a unit's target changes (primary units only)
- PLAYER_FOCUS_CHANGED is fired when the focus unit is changed
- UNIT_SPELLCAST_SENT, "player", spell, rank, target
- UNIT_SPELLCAST_START, unit
- UNIT_SPELLCAST_STOP, unit
- UNIT_SPELLCAST_FAILED, unit
- UNIT_SPELLCAST_INTERRUPTED, unit
- UNIT_SPELLCAST_DELAYED, unit
- UNIT_SPELLCAST_SUCCEEDED, "player", spell, rank
- UNIT_SPELLCAST_CHANNEL_START, unit
- UNIT_SPELLCAST_CHANNEL_UPDATE, unit
- UNIT_SPELLCAST_CHANNEL_STOP, unit
- The old SPELLCAST_* and SPELLCAST_CHANNEL_* events are obsolete.
- PLAYER_REGEN_DISABLED is sent before protected frames are locked down
- PLAYER_REGEN_ENABLED is sent after protected frames are unlocked
- MACRO_ACTION_FORBIDDEN: sent when a macro tries use actions that are always forbidden (movement, targeting, etc.). The first argument is the name of the protected function that was called.
- ADDON_ACTION_FORBIDDEN: sent when an addon tries use actions that are always forbidden (movement, targeting, etc.). The first argument is the name of the addon that was last involved in the execution path, and the second argument is the protected function that was called.
- MACRO_ACTION_BLOCKED: sent when a macro tries use actions that are blocked in combat or require interactive input (spell casting, trading, etc.). The first argument is the name of the protected function that was called.
- ADDON_ACTION_BLOCKED: sent when an addon tries use actions that are blocked in combat or require interactive input (spell casting, trading, etc.). The first argument is the name of the addon that was last involved in the execution path, and the second argument is the protected function that was called.
- MODIFIER_STATE_CHANGED, key, state: Sent whenever a modifier key's state changed, modifier is one of "SHIFT" "CTRL" or "ALT", state is 1 for down, 0 for up.
Protected Frame Features
- If the mouse is over a protected frame with the 'unit' attribute set, then the "mouseover" unit will refer to the frame's unit.
- Holding down ctrl, alt, shift, x will check keybindings in the following order: ALT-CTRL-SHIFT-X, CTRL-SHIFT-X, ALT-SHIFT-X, ALT-CTRL-X, SHIFT-X, CTRL-X, ALT-X, X *2.0.3*
- Key bindings that directly cast spells or use items now respect the self-cast modifier. *2.0.3*
- Arrays that are stored as saved variables are now saved using lua's array initializer syntax for optimal memory usage and performance on load.
Bug Fixes 2.0.1
- Fixed missing CHAT_MSG_SPELL_SELF_BUFF messages (Mining, Gathering), containers correctly do NOT fire the event.
Bug Fixes 2.0.3
- /click now works with secure action buttons
- Fixed taint issues with CloseLoot()
- Fixed taint issues with PickupContainerItem()
- MovePad will work again
- The raid pullout positions will *Still NOT* be saved and restored between sessions
- Fixed a bug where raid composition changes can incorrectly change the units referenced by the pullouts.
- Fixed taint issue with social frame/raid tab not showing in combat (buttons in main social/raid tab are no longer secure unit buttons)
- Fixed /reply in macros
- Fixed issue whereby creating Textures/FontStrings from a template used the template name instead of the specified region name.
- Fixed bug where /petattack would target the nearest unit, not nearest live enemy unit
Bug Fixes 2.0.4
- Fixed problem with name matching of players in the party or raid (affected promote, follow, etc.)
- Allow dueling option in tainted unit menus (moved up from 2.0.5 - Jan 11)
- Fixed /equip
Bug Fixes 2.0.6
- /petattack will treat target=pettarget, and /startattack will treat target=target the same way /focus treats target=focus
- Improved robustness of pet action bar display, now less likely to be accidentally blocked by addons.
- The RaidUI will no longer pop up a previously existing class window after it has been closed.
- Incessant gong sounds no longer play when you are in a Battleground/Arena queue.