[转]WOW原始UI代码分析【ActionButton】

[转]WOW原始UI代码分析【ActionButton】
http://blog.csdn.net/tcxxs/category/148474.aspx


1、这是我分析WOW原始UI代码后,对一些比较关键的部分,记录下来的
2、文章里面如果有什么纰漏,希望大家毫不客气的提出来,我一定查究!
3、关于基础问题可以参考
   WOW UI
《UI制作入门》http://www.pcgames.com.cn/netgames/zhuanti/wow/expert/Ui.htm
《WOW UI定制入门》http://202.113.13.169/site/mybbs/read.php?tid=33
《WOW插件制作指南》http://www.cnblogs.com/bluefee/archive/2005/04/14/137217.html
《ADDONS编写普及》http://wowbbs.game.mop.com/viewthread.php?tid=216755&extra=page%3D1
《动手写个属于自己的UI》http://wowbbs.game.mop.com/viewthread.php?tid=149814&fpage=1&highlight=
   XML
《XML初学进阶》http://www.chinausd.com/code/code.asp?id=173/5058
《XMLSpy 2005 Enterprise Edition》http://www.altova.com/download_spy_enterprise.html
   Lua
《Lua5.0参考手册中文版》http://www.cnblogs.com/bluefee/archive/2005/04/15/138576.html
《Lua 程序设计初步》http://bbs.battlecn.net/read.php?tid=541871&fpage=1
《通过例子学习Lua》http://www.wowtc.com/bbs/read.php?tid=4999&fpage=2
《Programming in LUA中文版》http://www.wowtc.com/bbs/read.php?tid=4998&fpage=2
《LuaEdit v2.5》http://luaforge.net/projects/luaedit/
4、关于如何提取MPQ文件,如何查看BLP图像,可以参考
《WoW Working Link》http://www.fukt.bth.se/%7Ek/wow/stuff/howto-extract-interface-files.txt
《Win MPQ》http://wow.duowan.com/2005-05-16/007R/26857843.html
《Win BLP Viewer》http://gamedown.yesky.com/game/108/108805.html
5、关于查询资料可以参考
《WoW Wiki》http://www.wowwiki.com/Interface_Customization/
《WOW APIs》http://www.cnblogs.com/bluefee/archive/2005/04/12/136270.html
《UI&Macro Forum》http://forums.worldofwarcraft.com/board.aspx?fn=wow-interface-customization
6、如果该问题是可以类推,或是从上下文能马上理解,或是从字面就知道含义的我就不罗嗦了
7、对于相同的问题我一般只分析一次,或是简要的提示一下,具体内容可以参考我以前的分析
8、如果你对自己查找资料的能力感到抱歉,可以E-mail向我询问,我会努力帮助你
9、联系方式E-mail: [email protected]    Blog: blog.csdn.net/tcxxs     QQ: 35548917
10、做人要厚道,转载请注明出处!


--------------------------------------------------


正式开始:
1、版本:1.7.1 (4659)
2、这次我写的是ActionbuttonTemplate.xml,ActionBarFrame.xml和ActionButton.lua
   他们是动作按钮的基础UI,也是大家比较关心的一部分,在Interface/FrameXML里面
3、这次分析用到的名词解释:
【模板】学过C++语言的人都知道虚函数,实际上是一个道理,只是作为子代继承的规范,不可创建实例
【栅格】一个用来放置图标的格子,默认为无图标时隐藏,当然你也可以显示它
【额外栅格】包括了战士姿态,德鲁依外形,盗贼潜行的栅格
【栅格ID】每个栅格都有对应的ID,是该栅格的唯一标示
【图标】每个技能和物品,包括交易技能,食物,宏等都有一个可以拖动的图标
【动作键】当把图标放置到栅格上时,这个整体就叫动作键
【额外动作键】当把图标放置到额外栅格上时,这个整体就叫额外动作键
【动作条】一般12个栅格组成一排或者一列,称这个为动作条
【额外动作条】由12个额外栅格组成一排或者一列,称这个为额外动作条
【动作条ID】在默认动作条中每个动作条都有页码,是这个动作条的唯一标示
  这里解释一下各个动作条的位置,主动作条包括1,2页
  然后右边1,右边2,右下,左下依次为3到6页,栅格ID排列如图
  61-72(左下)49-60(右下)37    25
        (右边2) I     I (右边1)
  1-12(主AB)13-24(2页)  48    36
  这里注意一个问题,为什么有的UI可以实现96个(FB,DUF等)或者更多呢
  这是因为额外栅格的存在,这些UI把空闲的额外栅格加以利用罢了
  调用这些额外栅格的动作键应该用其对应的函数
【闪动】当执行自动攻击或者自动射击技能时,改动作键就处于闪动状态
【CD】CoolDown的简写
【信息提示】即GameTooltip,当鼠标指向某个东西时显示的提示信息
【光环】包括buff,debuff,物品附加属性等
【arg1..N】arg为事件触发是传入的参数,可以为0到多个


--------------------------------------------

源代码及注释

-----------------------ActionButtonTemplate.xml


<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
..\FrameXML\UI.xsd">
<CheckButton name="ActionButtonTemplate" virtual="true">
【动作键】【模板】
  <Size>
   <AbsDimension x="36" y="36"/>
  </Size>
  <Layers>
   <Layer level="BACKGROUND">
    <Texture name="$parentIcon"/>
    背景层放置【图标】
   </Layer>
   <Layer level="ARTWORK">
   层的从下到上为:BackGround,ArtWork,OverLay
    <Texture name="$parentFlash" file="Interface\Buttons\UI-QuickslotRed" hidden="true"/>
    【闪动】效果的纹理
    <FontString name="$parentHotKey" inherits="NumberFontNormalSmallGray" justifyH="RIGHT">
    该【动作键】的快捷键信息
     <Size>
      <AbsDimension x="32" y="10"/>
     </Size>
     <Anchors>
      <Anchor point="TOPLEFT">
       <Offset>
        <AbsDimension x="2" y="-2"/>
       </Offset>
      </Anchor>
     </Anchors>
    </FontString>
    <FontString name="$parentCount" inherits="NumberFontNormal" justifyH="RIGHT">
    该【动作键】数量信息
     <Anchors>
     注意,这里没有Size标签,所以它为自动大小
      <Anchor point="BOTTOMRIGHT">
       <Offset>
        <AbsDimension x="-2" y="2"/>
       </Offset>
      </Anchor>
     </Anchors>
    </FontString>
   </Layer>
   <Layer level="OVERLAY">
    <FontString name="$parentName" inherits="GameFontHighlightSmallOutline">
    该【动作键】名字,例如:宏的名字
     <Size>
      <AbsDimension x="36" y="10"/>
     </Size>
     <Anchors>
      <Anchor point="BOTTOM">
       <Offset>
        <AbsDimension x="0" y="2"/>
       </Offset>
      </Anchor>
     </Anchors>
    </FontString>
   </Layer>
  </Layers>
  <Frames>
   <Model name="$parentCooldown" inherits="CooldownFrameTemplate"/>
   直接应用【CD】模板,没有测试过,还望高人赐教,或者等我以后试
  </Frames>
  <NormalTexture name="$parentNormalTexture" file="Interface\Buttons\UI-Quickslot2">
  【图标】边框纹理,如果该【栅格】已放置【图标】,则显示这个纹理
  若没有放置,则显示UI-Quickslot,它比UI-Quickslot2稍微小点
   <Size>
    <AbsDimension x="64" y="64"/>
   </Size>
   <Anchors>
    <Anchor point="CENTER">
     <Offset>
      <AbsDimension x="0" y="-1"/>
      立体效果,而且UI-Quickslot的显示位置也依赖UI-Quickslot2
      但是【图标】的实际放置位置却并不依赖这个位置,由BackGround层决定
     </Offset>
    </Anchor>
   </Anchors>
  </NormalTexture>
  <PushedTexture file="Interface\Buttons\UI-Quickslot-Depress"/>
  点击【动作键】时的纹理
  <HighlightTexture alphaMode="ADD" file="Interface\Buttons\ButtonHilight-Square"/>
  鼠标经过【栅格】时的纹理
  <CheckedTexture alphaMode="ADD" file="Interface\Buttons\CheckButtonHilight"/>
  等待施法时【栅格】的纹理,例如:你点击加血法术到选择目标那段时间显示
</CheckButton>
</Ui>


------------------------ActionBarFrame.xml


<Ui xmlns="http://www.blizzard.com/wow/ui/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.blizzard.com/wow/ui/
..\FrameXML\UI.xsd">
<Script file="ActionButton.lua"/>
<CheckButton name="ActionBarButtonTemplate" inherits="ActionButtonTemplate" virtual="true">
  <Scripts>
   <OnLoad>
    ActionButton_OnLoad();
   </OnLoad>
   <OnEvent>
    ActionButton_OnEvent(event);
   </OnEvent>
   <OnClick>
    if ( IsShiftKeyDown() ) then
    安住shift再点【动作键】,即拖拉【图标】
     PickupAction(ActionButton_GetPagedID(this));
     鼠标吸附该【图标】
    else
    否则执行该【动作键】命令
     MacroFrame_EditMacro();
     UseAction(ActionButton_GetPagedID(this), 1);
     UseAction(slot [,checkCursor] [,onSelf])函数
     slot参数为该【动作键】ID号
     checkCursor参数为是否在其他事件中返回是鼠标点击,0或者1
     onSelf参数为是否对自己施法,0或者1
    end
    ActionButton_UpdateState();
    点击后应该更新【动作键】状态
   </OnClick>
   <OnDragStart>
    if ( LOCK_ACTIONBAR ~= "1" ) then
    如果【动作键】并未设定为锁定
     PickupAction(ActionButton_GetPagedID(this));
     ActionButton_UpdateState();
    end
   </OnDragStart>
   <OnReceiveDrag>
    if ( LOCK_ACTIONBAR ~= "1" ) then
     PlaceAction(ActionButton_GetPagedID(this));
     PlaceAction,放下鼠标上吸附的【动作键】
     注意,这里并不要求一定要放在栅格上,因为你可以摧毁一个【动作键】
     ActionButton_UpdateState();
    end
   </OnReceiveDrag>
   <OnEnter>
    ActionButton_SetTooltip();
    经过【动作键】时,显示【信息提示】
   </OnEnter>
   <OnLeave>
    this.updateTooltip = nil;
    GameTooltip:Hide();
   </OnLeave>
   <OnUpdate>
    ActionButton_OnUpdate(arg1);
    更新【闪动】效果
   </OnUpdate>
  </Scripts>
</CheckButton>
下面定义了主【动作条】
<CheckButton name="ActionButton1" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="1">
MainMenuBarArtFrame,即默认最下面的一条工具栏,包括了主【动作条】,人物工具条,背包
但是注意,并不包括两旁的2只鹰的纹理
  <Anchors>
   <Anchor point="BOTTOMLEFT">
    <Offset>
     <AbsDimension x="8" y="4"/>
     左下和右下的【动作条】都依据主【动作条】的位置
     【信息提示】和【额外栅格】并不依据这个偏移,但是它会根据左下和右下【动作条】的出现而更改高度
     若仅改变该偏移,则如果该人物有【额外栅格】,则人物的主【动作条】将会在原位又出现一个
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton2" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="2">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton1" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton3" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="3">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton2" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton4" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="4">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton3" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton5" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="5">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton4" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton6" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="6">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton5" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton7" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="7">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton6" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton8" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="8">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton7" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton9" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="9">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton8" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton10" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="10">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton9" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton11" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="11">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton10" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<CheckButton name="ActionButton12" inherits="ActionBarButtonTemplate" parent="MainMenuBarArtFrame" id="12">
  <Anchors>
   <Anchor point="LEFT" relativeTo="ActionButton11" relativePoint="RIGHT">
    <Offset>
     <AbsDimension x="6" y="0"/>
    </Offset>
   </Anchor>
  </Anchors>
</CheckButton>
<Button name="ActionBarUpButton" parent="MainMenuBarArtFrame">
主【动作条】位置上的上翻页按钮
  <Size>
   <AbsDimension x="32" y="32"/>
  </Size>
  <Anchors>
   <Anchor point="CENTER" relativeTo="MainMenuBarArtFrame" relativePoint="TOPLEFT">
    <Offset>
     <AbsDimension x="522" y="-22"/>
    </Offset>
   </Anchor>
  </Anchors>
  <HitRectInsets>
   <AbsInset left="6" right="6" top="7" bottom="7"/>
   这个一直没有测试成功,还望高人指点迷津
  </HitRectInsets>
  <Scripts>
   <OnLoad>
    MainMenuBarPageNumber:SetText(CURRENT_ACTIONBAR_PAGE);
    CURRENT_ACTIONBAR_PAGE,当前【动作条ID】
    当你显示了其他【动作条】如:左下【动作条】,那么它将不会在翻页中重复出现
    this:RegisterEvent("ACTIONBAR_PAGE_CHANGED");
    当主【动作条】的页码改变时触发
   </OnLoad>
   <OnEvent>
    if ( event == "ACTIONBAR_PAGE_CHANGED" ) then
     MainMenuBarPageNumber:SetText(CURRENT_ACTIONBAR_PAGE);
    end
   </OnEvent>
   <OnClick>
    ActionBar_PageUp();
    PlaySound("UChatScrollButton");
    PlaySound(SoundName),播放一个WOW内置的声音
    SoundName为内置参数的名字,可以参考下面地址
    http://www.wowwiki.com/API_PlaySound
    注意,你不要试图在MPQ文件中找到这个名字
   </OnClick>
  </Scripts>
  <NormalTexture file="Interface\MainMenuBar\UI-MainMenu-ScrollUpButton-Up"/>
  <PushedTexture file="Interface\MainMenuBar\UI-MainMenu-ScrollUpButton-Down"/>
  <DisabledTexture file="Interface\Buttons\UI-ScrollBar-ScrollUpButton-Disabled"/>
  被锁定时的纹理,未测试成功,望高人指点一二
  <HighlightTexture alphaMode="ADD" file="Interface\MainMenuBar\UI-MainMenu-ScrollUpButton-Highlight"/>
</Button>
<Button name="ActionBarDownButton" parent="MainMenuBarArtFrame">
  <Size>
   <AbsDimension x="32" y="32"/>
  </Size>
  <Anchors>
   <Anchor point="CENTER" relativeTo="MainMenuBarArtFrame" relativePoint="TOPLEFT">
    <Offset>
     <AbsDimension x="522" y="-42"/>
    </Offset>
   </Anchor>
  </Anchors>
  <HitRectInsets>
   <AbsInset left="6" right="6" top="7" bottom="7"/>
  </HitRectInsets>
  <Scripts>
  上翻中已经注册并处理了事件,这里不重复
   <OnClick>
    ActionBar_PageDown();
    PlaySound("UChatScrollButton");
   </OnClick>
  </Scripts>
  <NormalTexture file="Interface\MainMenuBar\UI-MainMenu-ScrollDownButton-Up"/>
  <PushedTexture file="Interface\MainMenuBar\UI-MainMenu-ScrollDownButton-Down"/>
  <DisabledTexture file="Interface\Buttons\UI-ScrollBar-ScrollDownButton-Disabled"/>
  <HighlightTexture alphaMode="ADD" file="Interface\MainMenuBar\UI-MainMenu-ScrollDownButton-Highlight"/>
</Button>
</Ui>


------------------------ActionButton.lua

CURRENT_ACTIONBAR_PAGE = 1;
当前【动作条ID】
NUM_ACTIONBAR_PAGES = 6;
【动作条ID】总数
NUM_ACTIONBAR_BUTTONS = 12;
每个【动作条】的【动作键】数
ATTACK_BUTTON_FLASH_TIME = 0.4;
【闪动】效果时间间隔,为0.4S

BOTTOMLEFT_ACTIONBAR_PAGE = 6;
初始左下【动作条ID】
BOTTOMRIGHT_ACTIONBAR_PAGE = 5;
初始右下【动作条ID】
LEFT_ACTIONBAR_PAGE = 4;
初始右边2【动作条ID】
RIGHT_ACTIONBAR_PAGE = 3;
初始右边1【动作条ID】

-- Table of actionbar pages and whether they're viewable or not
VIEWABLE_ACTION_BAR_PAGES = {1, 1, 1, 1, 1, 1};
初始化为每页【动作条】都能够显示,0为隐藏

function ActionButtonDown(id)
实现按下【动作键】
if ( BonusActionBarFrame:IsVisible() ) then
如果是【额外动作键】
  local button = getglobal("BonusActionButton"..id);
  if ( button:GetButtonState() == "NORMAL" ) then
  若未按下
   button:SetButtonState("PUSHED");
  end
  return;
  该函数用if-else是同样的
end

local button = getglobal("ActionButton"..id);
if ( button:GetButtonState() == "NORMAL" ) then
  button:SetButtonState("PUSHED");
end
end

由上下2个函数可以看出,暴雪对【动作键】和【额外动作键】同样对待
这个也是其他UI可以利用【额外栅格】的基础

function ActionButtonUp(id, onSelf)
if ( BonusActionBarFrame:IsVisible() ) then
  local button = getglobal("BonusActionButton"..id);
  if ( button:GetButtonState() == "PUSHED" ) then
   button:SetButtonState("NORMAL");
   -- Used to save a macro
   MacroFrame_EditMacro();
   UseAction(ActionButton_GetPagedID(button), 0);
   if ( IsCurrentAction(ActionButton_GetPagedID(button)) ) then
   IsCurrentAction,等待施法,并不是正在吟唱
    button:SetChecked(1);
    设定该【动作键】正处于等待状态
   else
    button:SetChecked(0);
   end
  end
  return;
end

local button = getglobal("ActionButton"..id);
if ( button:GetButtonState() == "PUSHED" ) then
  button:SetButtonState("NORMAL");
  -- Used to save a macro
  MacroFrame_EditMacro();
  UseAction(ActionButton_GetPagedID(button), 0, onSelf);
  if ( IsCurrentAction(ActionButton_GetPagedID(button)) ) then
   button:SetChecked(1);
  else
   button:SetChecked(0);
  end
end
end

function ActionBar_PageUp()
CURRENT_ACTIONBAR_PAGE = CURRENT_ACTIONBAR_PAGE + 1;
local nextPage;
for i=CURRENT_ACTIONBAR_PAGE, NUM_ACTIONBAR_PAGES do
  if ( VIEWABLE_ACTION_BAR_PAGES[i] ) then
  如果该页能显示
   nextPage = i;
   break;
  end
end

if ( not nextPage ) then
如果没有找到能够显示的下一页
  CURRENT_ACTIONBAR_PAGE = 1;
else
  CURRENT_ACTIONBAR_PAGE = nextPage;
end
ChangeActionBarPage();
这个函数变换将当主【动作条】变换到CURRENT_ACTIONBAR_PAGE指向的【动作条】
end

function ActionBar_PageDown()
CURRENT_ACTIONBAR_PAGE = CURRENT_ACTIONBAR_PAGE - 1;
local prevPage;
for i=CURRENT_ACTIONBAR_PAGE, 1, -1 do
  if ( VIEWABLE_ACTION_BAR_PAGES[i] ) then
   prevPage = i;
   break;
  end
end

if ( not prevPage ) then
  for i=NUM_ACTIONBAR_PAGES, 1, -1 do
  把“1”换成“CURRENT_ACTIONBAR_PAGE+1”也是可以的
  把这个for循环对比ActionBar_PageUp的同一地方
  可见,暴雪员工相当的肯定在上翻的时候第1页是可以显示的
  而在下翻的时候显得信心不足,还重复检查了一次NUM_ACTIONBAR_PAGES到CURRENT_ACTIONBAR_PAGE
  这是因为左下、右下AB的出现只是修改5,6页的可见性,并不影响1页的可见性
  所以在下翻的时候需要对1到6页均做可见性检查
  但是,重复检查的区间的确是多余的,估计是暴雪员工偷懒
  因为“1”比“CURRENT_ACTIONBAR_PAGE+1”要好写得多
   if ( VIEWABLE_ACTION_BAR_PAGES[i] ) then
    prevPage = i;
    break;
   end
  end
end
CURRENT_ACTIONBAR_PAGE = prevPage;
ChangeActionBarPage();
end

function ActionButton_OnLoad()
this.showgrid = 0;
显示空【栅格】的标志
this.flashing = 0;
处于【闪动】的标志
this.flashtime = 0;
【闪动】时间间隔计算的中间量
ActionButton_Update();
this:RegisterForDrag("LeftButton", "RightButton");
对this所指向的窗体注册鼠标拖拉行为,参数至少一个,也可以注册多个
参数为LeftButton,RightButton,表示使用某个鼠标键执行
this:RegisterForClicks("LeftButtonUp", "RightButtonUp");
对this所指向的窗体注册鼠标点击行为,参数至少一个,也可以注册多个
参数为LeftButtonDown,LeftButtonUp,RightButtonDown,RightButtonUp
注意,在OnClick标签中传入的arg1参数只返回LeftButton或者RightButton
this:RegisterEvent("PLAYER_ENTERING_WORLD");
this:RegisterEvent("UPDATE_BONUS_ACTIONBAR");
当【额外栅格】更新时触发
this:RegisterEvent("ACTIONBAR_SHOWGRID");
当拖拉【图标】时触发,arg1为LeftButton或者RightButton
this:RegisterEvent("ACTIONBAR_HIDEGRID");
当停止拖拉【图标】时触发
this:RegisterEvent("ACTIONBAR_PAGE_CHANGED");
this:RegisterEvent("ACTIONBAR_SLOT_CHANGED");
当【栅格】内容变更时触发,arg1为变更【栅格ID】
this:RegisterEvent("ACTIONBAR_UPDATE_STATE");
当【栅格】状态更新时触发,包括了【CD】和可用性鉴定
但是不包括自动攻击和自动射击事件
arg1为鼠标LeftButton,RightButton,也可能为nil
this:RegisterEvent("ACTIONBAR_UPDATE_USABLE");
当更新【图标】可用性后触发
this:RegisterEvent("ACTIONBAR_UPDATE_COOLDOWN");
当更新【CD】后触发
当【CD】开始的时候,arg1返回使用的鼠标按钮LeftButton或者RightButton
当【CD】结束的时候,arg1返回nil
this:RegisterEvent("UPDATE_INVENTORY_ALERTS");
当变更装备时或者经过一定时间后触发
this:RegisterEvent("PLAYER_AURAS_CHANGED");
当玩家【光环】改变时触发,包括了【光环】的消失和出现
注意,属性完全相同【光环】重复释放并不能触发该事件
this:RegisterEvent("PLAYER_TARGET_CHANGED");
当玩家的目标变更时触发,包括取消和选定目标
arg1为该目标持续时间,单位分钟,小数保留3位
this:RegisterEvent("UNIT_AURASTATE");
当某单位短时间【光环】改变时触发,需进一步测试
this:RegisterEvent("UNIT_INVENTORY_CHANGED");
当玩家装备变更时或者当观察玩家装备并发生装备变更时触发
this:RegisterEvent("CRAFT_SHOW");
this:RegisterEvent("CRAFT_CLOSE");
工艺窗口的显示与关闭,arg1为鼠标点击键
this:RegisterEvent("TRADE_SKILL_SHOW");
this:RegisterEvent("TRADE_SKILL_CLOSE");
交易技能制作物品窗口的显示与关闭,arg1为鼠标点击键
this:RegisterEvent("UNIT_HEALTH");
当任何一个单位的血量发生变化时触发,arg1参数为该单位ID
player,pet,target,mouseover,party1..4,partypet1..4,raid1..40,raidpet1..40,npc
注意,该事件拥有很高的优先级,特别当参数为player时
this:RegisterEvent("UNIT_MANA");
当任何一个单位的魔法量发生变化时触发,arg1参数为该单位ID
this:RegisterEvent("UNIT_RAGE");
当任何一个单位的怒气值发生变化时触发,arg1参数为该单位ID
this:RegisterEvent("UNIT_FOCUS");
当任何一个单位的焦距值发生变化时触发,arg1参数为该单位ID
注意,该事件实际上已经废除,因为现在猎人使用MANA,而不是FOCUS
this:RegisterEvent("UNIT_ENERGY");
当任何一个单位的能量发生变化时触发,arg1参数为该单位ID
this:RegisterEvent("PLAYER_ENTER_COMBAT");
当玩家进入自动攻击状态时触发,自动射击不包括在内
注意,仅当你使用自动攻击时触发该事件
使用其他任何技能或者攻击行为都不可触发
this:RegisterEvent("PLAYER_LEAVE_COMBAT");
离开自动攻击状态或者死亡时触发
this:RegisterEvent("PLAYER_COMBO_POINTS");
当连击点变更时触发,arg1总返回player,而并不时连击点数
this:RegisterEvent("UPDATE_BINDINGS");
当快捷键更改时触发
this:RegisterEvent("START_AUTOREPEAT_SPELL");
当自动射击开始时触发,而不是自动攻击
this:RegisterEvent("STOP_AUTOREPEAT_SPELL");
离开自动射击状态或者死亡时触发
ActionButton_UpdateHotkeys();
end

function ActionButton_UpdateHotkeys(actionButtonType)
if ( not actionButtonType ) then
  actionButtonType = "ACTIONBUTTON";
end
local hotkey = getglobal(this:GetName().."HotKey");
该对象在模板中为Fontstring
local action = actionButtonType..this:GetID();
actionButtonType,为该类命令的名字
this:GetID,得到该命令在该类中的ID号
注意,这2个字符串组成了最终的命令的名字
如NUMPAD2,NUMPAD为类型,2为ID
但是有些类型就一个命令无ID,如LEFT
hotkey:SetText(KeyBindingFrame_GetLocalizedName(GetBindingKey(action), "KEY_"));
GetBindingKey得到命令的快捷键名字,后面加上KEY_是为了显示命令的字串
例如:KEY_NUMPAD2就显示为“数字键盘2”
KeyBindingFrame_GetLocalizedName,由这个函数在快捷键窗口中显示该命令的快捷键
end

function ActionButton_Update()
-- Special case code for bonus bar buttons
-- Prevents the button from updating if the bonusbar is still in an animation transition
if ( this.isBonus and this.inTransition ) then
  ActionButton_UpdateUsable();
  ActionButton_UpdateCooldown();
  return;
end

local icon = getglobal(this:GetName().."Icon");
local buttonCooldown = getglobal(this:GetName().."Cooldown");
local texture = GetActionTexture(ActionButton_GetPagedID(this));
if ( texture ) then
  icon:SetTexture(texture);
  icon:Show();
  如果该【栅格】有【图标】,即若为【动作键】则显示
  this.rangeTimer = TOOLTIP_UPDATE_TIME;
  每隔TOOLTIP_UPDATE_TIME事件间隔更新一次距离信息
  TOOLTIP_UPDATE_TIME为0.2S
  this:SetNormalTexture("Interface\\Buttons\\UI-Quickslot2");
  -- Save texture if the button is a bonus button, will be needed later
  if ( this.isBonus ) then
  判断自己是否为【额外栅格】
   this.texture = texture;
  end
else
  icon:Hide();
  buttonCooldown:Hide();
  this.rangeTimer = nil;
  this:SetNormalTexture("Interface\\Buttons\\UI-Quickslot");
  getglobal(this:GetName().."HotKey"):SetVertexColor(0.6, 0.6, 0.6);
end
ActionButton_UpdateCount();
if ( HasAction(ActionButton_GetPagedID(this)) ) then
判断该【栅格】是否有【图标】,即是否为【动作键】
  this:Show();
  ActionButton_UpdateState();
  ActionButton_UpdateUsable();
  ActionButton_UpdateCooldown();
  ActionButton_UpdateFlash();
elseif ( this.showgrid == 0 ) then
  this:Hide();
else
这里表示,当该【栅格】没有【图标】
但是显示空【栅格】选项开启时,不显示公共CD效果
  buttonCooldown:Hide();
end
if ( GameTooltip:IsOwned(this) ) then
判断该【动作键】是否拥有【信息提示】
  ActionButton_SetTooltip();
else
  this.updateTooltip = nil;
end

-- Update Macro Text
local macroName = getglobal(this:GetName().."Name");
macroName:SetText(GetActionText(ActionButton_GetPagedID(this)));
GetActionText返回宏的名字
end

function ActionButton_ShowGrid(button)
if ( not button ) then
  button = this;
end
button.showgrid = button.showgrid+1;
getglobal(button:GetName().."NormalTexture"):SetVertexColor(1.0, 1.0, 1.0, 0.5);
SetVertexColor(R,G,B)这个函数需要注意,可能学过3D的都知道,这个是顶点着色,取值0到1小数
顶点着色并不是直接设置RGB的数值,而是在原有的RGB数值上乘以顶点着色指定的RGB百分比修正值
例如:一个色块RGB为50,100,30,如果你写入SetVertexColor(0.5,0,1),结果RGB为25,0,30
button:Show();
end

function ActionButton_HideGrid(button)
if ( not button ) then
  button = this;
end
button.showgrid = button.showgrid-1;
if ( button.showgrid == 0 and not HasAction(ActionButton_GetPagedID(button)) ) then
把这里判断showgrid变量为0和这一对函数对showgrid的操作联系起来看
就像是用堆栈实现括号配对一样保证了显示和隐藏的规范
  button:Hide();
end
end

function ActionButton_UpdateState()
if ( IsCurrentAction(ActionButton_GetPagedID(this)) or IsAutoRepeatAction(ActionButton_GetPagedID(this)) ) then
IsAutoRepeatAction,自动射击
IsAttackAction,自动攻击
  this:SetChecked(1);
else
  this:SetChecked(0);
end
end

function ActionButton_UpdateUsable()
注意,这个函数的主体是个技能按钮模板,所以针对的是一个【动作键】
所以,当你拖动某个【图标】时,其他【图标】并不会跟着更新
local icon = getglobal(this:GetName().."Icon");
模板中定义
local normalTexture = getglobal(this:GetName().."NormalTexture");
local isUsable, notEnoughMana = IsUsableAction(ActionButton_GetPagedID(this));
得到当前技能栅格的ID,这个是所有栅格ID号,包括73~96的
if ( isUsable ) then
是否在射程,当前在其他游戏规则中是否可用,是否在CD等
  icon:SetVertexColor(1.0, 1.0, 1.0);

  normalTexture:SetVertexColor(1.0, 1.0, 1.0);
elseif ( notEnoughMana ) then
是否足够魔法、怒气、能量
  icon:SetVertexColor(0.5, 0.5, 1.0);
  normalTexture:SetVertexColor(0.5, 0.5, 1.0);
else
  icon:SetVertexColor(0.4, 0.4, 0.4);
  这个是灰色效果,当你拖拉【图标】时,原【栅格】则为这个效果
  normalTexture:SetVertexColor(1.0, 1.0, 1.0);
end
end

function ActionButton_UpdateCount()
local text = getglobal(this:GetName().."Count");
在模板中数量为Fontstring
if ( IsConsumableAction(ActionButton_GetPagedID(this)) ) then
判断是否改【栅格】的物品为消费品
  text:SetText(GetActionCount(ActionButton_GetPagedID(this)));
else
  text:SetText("");
end
end

function ActionButton_UpdateCooldown()
local cooldown = getglobal(this:GetName().."Cooldown");
在模板中该【CD】定义为一个Model
local start, duration, enable = GetActionCooldown(ActionButton_GetPagedID(this));
CooldownFrame_SetTimer(cooldown, start, duration, enable);
end

function ActionButton_OnEvent(event)
if ( event == "ACTIONBAR_SLOT_CHANGED" ) then
  if ( arg1 == -1 or arg1 == ActionButton_GetPagedID(this) ) then
  如果变更的【栅格】不明或者变更的【栅格】为自己则更新
   ActionButton_Update();
  end
  return;
end
if ( event == "PLAYER_ENTERING_WORLD" or event == "ACTIONBAR_PAGE_CHANGED" ) then
总结起来是,当游戏环境有大的变化时,更新自己
  ActionButton_Update();
  return;
end
if ( event == "UPDATE_BONUS_ACTIONBAR" ) then
  if ( this.isBonus ) then
   ActionButton_Update();
  end
  return;
end
if ( event == "ACTIONBAR_SHOWGRID" ) then
  ActionButton_ShowGrid();
  return;
end
if ( event == "ACTIONBAR_HIDEGRID" ) then
  ActionButton_HideGrid();
  return;
end
if ( event == "UPDATE_BINDINGS" ) then
  ActionButton_UpdateHotkeys();
  return;
end

-- All event handlers below this line MUST only be valid when the button is visible
if ( not this:IsVisible() ) then
如果该【图标】不可见,那么直接返回,即下面的所有事件均要求该【图标】可见,即应该为一个【动作键】
  return;
end

if ( event == "UNIT_HEALTH" or event == "UNIT_MANA" or event == "UNIT_RAGE" or event == "UNIT_FOCUS" or event == "UNIT_ENERGY" ) then
  if ( arg1 == "player" ) then
  如果发生在玩家身上,应该判断所有技能的可用性
   ActionButton_UpdateUsable();
  end
elseif ( event == "PLAYER_TARGET_CHANGED" or event == "PLAYER_AURAS_CHANGED" ) then
  ActionButton_UpdateUsable();
elseif ( event == "UNIT_AURASTATE" ) then
  if ( arg1 == "player" or arg1 == "target" ) then
   ActionButton_UpdateUsable();
  end
elseif ( event == "UNIT_INVENTORY_CHANGED" ) then
  if ( arg1 == "player" ) then
   ActionButton_Update();
  end
elseif ( event == "ACTIONBAR_UPDATE_STATE" ) then
  ActionButton_UpdateState();
elseif ( event == "ACTIONBAR_UPDATE_USABLE" or event == "UPDATE_INVENTORY_ALERTS" or event == "ACTIONBAR_UPDATE_COOLDOWN" ) then
  ActionButton_UpdateUsable();
  ActionButton_UpdateCooldown();
elseif ( event == "CRAFT_SHOW" or event == "CRAFT_CLOSE" or event == "TRADE_SKILL_SHOW" or event == "TRADE_SKILL_CLOSE" ) then
  ActionButton_UpdateState();
elseif ( event == "PLAYER_ENTER_COMBAT" ) then
  if ( IsAttackAction(ActionButton_GetPagedID(this)) ) then
   ActionButton_StartFlash();
  end
elseif ( event == "PLAYER_LEAVE_COMBAT" ) then
  if ( IsAttackAction(ActionButton_GetPagedID(this)) ) then
   ActionButton_StopFlash();
  end
elseif ( event == "PLAYER_COMBO_POINTS" ) then
  ActionButton_UpdateUsable();
elseif ( event == "START_AUTOREPEAT_SPELL" ) then
  if ( IsAutoRepeatAction(ActionButton_GetPagedID(this)) ) then
   ActionButton_StartFlash();
  end
elseif ( event == "STOP_AUTOREPEAT_SPELL" ) then
  if ( ActionButton_IsFlashing() and not IsAttackAction(ActionButton_GetPagedID(this)) ) then
  如果该技能在【闪动】,并且不是自动攻击
   ActionButton_StopFlash();
  end
end
end

function ActionButton_SetTooltip()
if ( GetCVar("UberTooltips") == "1" ) then
Uber,乳房。。。。。
这句应该是判断是否属于一般类型,母类型
UberTooltips详细信息提示
  GameTooltip_SetDefaultAnchor(GameTooltip, this);
  一般位于屏幕右下角,会根据【动作条】的多少自动改变位置
else
如果这个【栅格】的【信息提示】不属于默认的形式则
  if ( this:GetParent() == MultiBarBottomRight or this:GetParent() == MultiBarRight or this:GetParent() == MultiBarLeft ) then
  【动作条】MultiBarBottomRight右下,MultiBarRight右方,MultiBarLeft左边下
   GameTooltip:SetOwner(this, "ANCHOR_LEFT");
   这里解释下SetOwner函数,GameTooltip:SetOwner(owner, anchor);
   其中owner是参照物,anchor是对齐方式,取值如下:
   ANCHOR_TOPRIGHT 相当于 SetPoint("BOTTOMRIGHT",object,"TOPRIGHT")
   ANCHOR_RIGHT 相当于 SetPoint("BOTTOMLEFT",object,"TOPRIGHT")
   ANCHOR_BOTTOMRIGHT 相当于 SetPoint("TOPLEFT",object,"BOTTOMRIGHT")
   ANCHOR_TOPLEFT 相当于 SetPoint("BOTTOMLEFT",object,"TOPLEFT")
   ANCHOR_LEFT 相当于 SetPoint("BOTTOMRIGHT",object,"TOPLEFT")
   ANCHOR_BOTTOMLEFT 相当于 SetPoint("TOPRIGHT",object,"BOTTOMLEFT")
  else
   GameTooltip:SetOwner(this, "ANCHOR_RIGHT");
  end
end

if ( GameTooltip:SetAction(ActionButton_GetPagedID(this)) ) then
SetAction,返回值为该【信息提示】是否需要实时更新
  this.updateTooltip = TOOLTIP_UPDATE_TIME;
else
  this.updateTooltip = nil;
end
end

function ActionButton_OnUpdate(elapsed)
if ( ActionButton_IsFlashing() ) then
  this.flashtime = this.flashtime - elapsed;
  if ( this.flashtime <= 0 ) then
  flashtime初始为0,即第一次flash的时间并不为0.4,而是第一个elapsed决定
   local overtime = -this.flashtime;
   overtime,该函数的一个中间量,用语变化符号
   if ( overtime >= ATTACK_BUTTON_FLASH_TIME ) then
   这个if为了下面给flashtime更新是不出现负数考虑
   注意,ERROR文本框的文本并不和这里的时间挂钩
    overtime = 0;
   end
   this.flashtime = ATTACK_BUTTON_FLASH_TIME - overtime;
   更新flashtime,使得从第二次开始间隔一个ATTACK_BUTTON_FLASH_TIME

   local flashTexture = getglobal(this:GetName().."Flash");
   if ( flashTexture:IsVisible() ) then
   效果的实现即为反复显示这个flash纹理
    flashTexture:Hide();
   else
    flashTexture:Show();
   end
  end
end

-- Handle range indicator
if ( this.rangeTimer ) then
  if ( this.rangeTimer < 0 ) then
  
   local count = getglobal(this:GetName().."HotKey");
   if ( IsActionInRange( ActionButton_GetPagedID(this)) == 0 ) then
   IsActionInRange,返回指定【动作键】是否在距离之内
   返回nil表示该【栅格】无【图标】,或者没有选定目标
   返回0表示不在距离之内,返回1表示在距离之内
   注意,如果你没有试图释放这个【动作键】在目标身上,它可能总返回1
    count:SetVertexColor(1.0, 0.1, 0.1);
   else
    count:SetVertexColor(0.6, 0.6, 0.6);
   end
   this.rangeTimer = TOOLTIP_UPDATE_TIME;
  else
   this.rangeTimer = this.rangeTimer - elapsed;
  end
end

if ( not this.updateTooltip ) then
  return;
end

this.updateTooltip = this.updateTooltip - elapsed;
if ( this.updateTooltip > 0 ) then
  return;
end

if ( GameTooltip:IsOwned(this) ) then
  ActionButton_SetTooltip();
else
  this.updateTooltip = nil;
end
end

function ActionButton_GetPagedID(button)
if( button == nil ) then
  message("nil button passed into ActionButton_GetPagedID(), contact Jeff");
  严重错误需要弹出消息框
  return 0;
end
if ( button.isBonus and CURRENT_ACTIONBAR_PAGE == 1 ) then
如果是【额外动作条】并且当前【动作条ID】为1
  local offset = GetBonusBarOffset();
  GetBonusBarOffset,得到【额外动作条】的偏移量
  返回1到3,分别对应1到3号战士姿态,或者德鲁依外形,或者是盗贼状态
  if ( offset == 0 and BonusActionBarFrame and BonusActionBarFrame.lastBonusBar ) then
  应该是判断当意外失去【额外动作条】定位时,定位于最近一次改变的
  注意,如果该职业无【额外动作条】时,偏移量也为0
   offset = BonusActionBarFrame.lastBonusBar;
  end
  return (button:GetID() + ((NUM_ACTIONBAR_PAGES + offset - 1) * NUM_ACTIONBAR_BUTTONS));
  注意NUM_ACTIONBAR_PAGES + offset用的是加法,所以我们肯定暴雪偷偷藏了24个【额外栅格】
elseif ( button:GetParent():GetName() == "MultiBarBottomLeft" ) then
  return (button:GetID() + ((BOTTOMLEFT_ACTIONBAR_PAGE - 1) * NUM_ACTIONBAR_BUTTONS));
elseif ( button:GetParent():GetName() == "MultiBarBottomRight" ) then
  return (button:GetID() + ((BOTTOMRIGHT_ACTIONBAR_PAGE - 1) * NUM_ACTIONBAR_BUTTONS));
elseif ( button:GetParent():GetName() == "MultiBarLeft" ) then
  return (button:GetID() + ((LEFT_ACTIONBAR_PAGE - 1) * NUM_ACTIONBAR_BUTTONS));
elseif ( button:GetParent():GetName() == "MultiBarRight" ) then
  return (button:GetID() + ((RIGHT_ACTIONBAR_PAGE - 1) * NUM_ACTIONBAR_BUTTONS));
else
  return (button:GetID() + ((CURRENT_ACTIONBAR_PAGE - 1) * NUM_ACTIONBAR_BUTTONS))
end
end

function ActionButton_UpdateFlash()
local pagedID = ActionButton_GetPagedID(this);
if ( (IsAttackAction(pagedID) and IsCurrentAction(pagedID)) or IsAutoRepeatAction(pagedID) ) then
  ActionButton_StartFlash();
else
  ActionButton_StopFlash();
end
end

function ActionButton_StartFlash()
this.flashing = 1;
this.flashtime = 0;
ActionButton_UpdateState();
end

function ActionButton_StopFlash()
this.flashing = 0;
getglobal(this:GetName().."Flash"):Hide();
ActionButton_UpdateState();
end

function ActionButton_IsFlashing()
if ( this.flashing == 1 ) then
  return 1;
else
  return nil;
end
end


---------------------总结一下


常量
TOOLTIP_UPDATE_TIME         【信息提示】更新间隔
CURRENT_ACTIONBAR_PAGE        当前【动作条ID】
NUM_ACTIONBAR_PAGES         总页码
NUM_ACTIONBAR_BUTTONS        一个【动作条】的【栅格】数
ATTACK_BUTTON_FLASH_TIME       【闪动】间隔
BOTTOMLEFT_ACTIONBAR_PAGE       左下【动作条ID】
BOTTOMRIGHT_ACTIONBAR_PAGE       右下【动作条ID】
LEFT_ACTIONBAR_PAGE         右边2【动作条ID】
RIGHT_ACTIONBAR_PAGE        右边1【动作条ID】

事件
ACTIONBAR_PAGE_CHANGED        【动作条】翻页
UPDATE_BONUS_ACTIONBAR        【额外动作条】更新
ACTIONBAR_SHOWGRID         显示【栅格】
ACTIONBAR_HIDEGRID         隐藏【栅格】
ACTIONBAR_SLOT_CHANGED        【栅格】内容变更
ACTIONBAR_UPDATE_STATE        【动作键】更新状态
ACTIONBAR_UPDATE_USABLE        【动作键】更新可用性
ACTIONBAR_UPDATE_COOLDOWN       【动作键】更新【CD】
UPDATE_INVENTORY_ALERTS        装备警告
PLAYER_AURAS_CHANGED        【光环】改变
PLAYER_TARGET_CHANGED        目标改变
UNIT_AURASTATE          单位【光环】改变
UNIT_INVENTORY_CHANGED        单位装备改变
CRAFT_SHOW           工艺窗口显示
CRAFT_CLOSE           工艺窗口隐藏
TRADE_SKILL_SHOW         交易技能窗口显示
TRADE_SKILL_CLOSE         交易技能窗口隐藏
UNIT_HEALTH           单位血量改变(优先级极高)
UNIT_MANA           单位魔法改变
UNIT_RAGE           单位怒气改变
UNIT_FOCUS           单位焦距改变(已废除)
UNIT_ENERGY           单位能量改变
PLAYER_ENTER_COMBAT         开始自动攻击
PLAYER_LEAVE_COMBAT         结束自动攻击
PLAYER_COMBO_POINTS         连击点改变
UPDATE_BINDINGS          快捷键更新
START_AUTOREPEAT_SPELL        开始自动射击
STOP_AUTOREPEAT_SPELL        结束自动射击

函数
IsShiftKeyDown          Shift键是否按下
PickupAction          吸附【图标】
PlaceAction           放置【图标】
UseAction           使用【动作键】
GetButtonState          得到【动作键】状态
SetButtonState          设置【动作键】状态
Frame:RegisterForDrag        注册拖拉行为
Frame:RegisterForClicks        注册点击行为
GetBindingKey          得到快捷键
HasAction           【栅格】是否放置了【图标】
Frame:SetVertexColor        顶点色彩修正
IsAutoRepeatAction         是否正在自动射击
IsAttackAction          是否正在自动攻击
IsUsableAction          【动作键】是否可用
IsConsumableAction         是否为消费品
GetActionCooldown         得到【CD】
GetCVar            有待测试
GameTooltip:SetAction        设置【信息提示】
Frame:IsVisible          是否可见
IsActionInRange          是否在距离之内
GetBonusBarOffset         得到【额外动作条】偏移量



累死了。。。。。。明天再贴图片和排版彩色

http://wowbbs.game.mop.com/forumdisplay.php?fid=138

这个帖子我已经贴了图片

你可能感兴趣的:([转]WOW原始UI代码分析【ActionButton】)