Flash的事件機制

1.先說一些基礎的東西(直接從網上copy過來)

(1) 事件是一個對象,它由flash play產生和管理。事件流分捕獲階段,目標階段和冒泡階段。事件流的示意圖如下,這裏說明一下target和 currentTarget的概念。當一個事件流產生時,就產生了一個顯示對象列表,而target是最裏層的對象,而 currentTarget是捕獲階段和冒泡階段被"遍歷"的顯示對象,當然它只有註冊了時間監聽器,我們才能獲得 event.currentTarget。


(2)

Java代碼 複製代碼
  1. <1> 捕獲階段 (EventPhase.CAPTURING_PHASE)包括從舞臺到目標節點範圍內的所有節點   
  2.  <2> 目標階段 (EventPhase.AT_TARGET)權包括目標節點   
  3.  <3> 冒泡階段 (EventPhase.BUBBLING_PHASE)從目標節點的父節點返回到舞臺的行程中遇到的節點  
   <1> 捕獲階段 (EventPhase.CAPTURING_PHASE)包括從舞臺到目標節點範圍內的所有節點
    <2> 目標階段 (EventPhase.AT_TARGET)權包括目標節點
    <3> 冒泡階段 (EventPhase.BUBBLING_PHASE)從目標節點的父節點返回到舞臺的行程中遇到的節點



(3)不是所有的事件都有這三個階段。如Timer、URLLoader,它們的事件對象將直接派送給目標對象(target).它們只包含目標階段而沒有捕獲階段和冒泡階段。它們不會像顯示對象容器(DisplayObjectContainer)那樣有可能被一個DisplayObjectContainer對象包含或者自己包含一個DisplayObjectContainer對象,它們往往是單獨存在的。也就是說如果一個顯示對象不在顯示列表中時,FlashPlaye會把事件直接派送給它,這個時候就沒有事件流,也沒有捕獲階段和冒泡階段,只有目標階段。





(1)請仔細看上面的圖中的事件流,圖中的假定的事件派發的對象是子節點1,從圖中我們可以看出,事件流並沒有經過子節點2,也就是說如果子節點2中監聽事件,那子節點無法監聽到事件。
(2)如果如果父節點監聽子節點1的事件可以監督到嗎?當然可以因爲事件流有經過父節點;有兩種方式可以監聽到子節點1的事件:
  自定義事件:
    1)在父節點裏面用子節點監聽事件(如子節點1.addEventListener(。。。。))此時不需要設置addEventListener中的useCapture參數和new Event時Event的bubbles參數。就可以直接監聽到子節點1的事件,因爲此時直接在目標階段捕捉事件。
    2)直接用在父節點中加入但是addEventListener(。。。)也可以監聽子節點1的事件,但是前提條件是在父節點監聽事件的時候要把addEventListener中的useCapture參數設置成true(如addEventListener(MenuEvent.MENU_Event, handler,true)或者字節點1中new Event時將Event的bubbles參數設置爲true( new MenuEvent(MenuEvent.MENU_Event,true))。如果都不這樣的話,父節點是無法監聽到事件的。



因爲addEventListener中的useCapture的default模式是false也就是偵聽器只在目標或冒泡階段處理事件,而new Event時將Event的bubbles參數default也是false也就是說不參與事件流的冒泡階段。如果不設置useCapture或者bubbles爲true那父節點肯定無法監聽到事件。
(3)如果非要在子節點2中監聽子節點1中發出的事件怎麼辦?
    1)辦法有很多種,說一下最常用的一種,就是向子節點1 與子節點2 都傳入一個相同的對象的實例,在子節點1中用這個對象實例發出事件,在子節點2中用這個對象實例監聽事件(在目標階段捕獲事件)。這樣就可以收到字節點1的事件了。
  非自定義事件(如mouseEvent):
   1)當是非自定義事件時。直接用在父節點中加入但是addEventListener(。。。)就可以監聽子節點1的事件。
   2)有人可能要問如果我要在子節點2監聽子節點1的非自定義事件呢?可以用(上面的2 標題裏面 3)),但是完全沒有必要,比如,你說你在子節點2中監聽子節點1的mouseClick 事件幹嗎??


摘錄Flash幫助文檔中的規範解釋:

addEventListener () 方法

public function addEventListener(type:String, listener:Function, useCapture:Boolean = false, priority:int = 0, useWeakReference:Boolean = false):void

語言版本 :  ActionScript 3.0
RuntimeVersions:  AIR 1.0, Flash Player 9



使用 EventDispatcher 對象註冊事件偵聽器對象,以使偵聽器能夠接收事件通知。可以爲特定類型的事件、階段和優先級在顯示列表的所有節點上註冊事件偵聽器。

成功註冊一個事件偵聽器後,無法通過額外調用 addEventListener() 來更改其優先級。要更改偵聽器的優先級,必須首先調用 removeListener()。然後,可以使用新的優先級再次註冊該偵聽器。

請記住,註冊該偵聽器後,如果繼續調用具有不同 type 或 useCapture 值的 addEventListener(),則會創建單獨的偵聽器註冊。例如,如果首先註冊 useCapture 設置爲 true 的偵聽器,則該偵聽器只在捕獲階段進行偵聽。如果使用同一個偵聽器對象再次調用 addEventListener(),並將 useCapture 設置爲 false,那麼便會擁有兩個單獨的偵聽器:一個在捕獲階段進行偵聽,另一個在目標和冒泡階段進行偵聽。

不能只爲目標階段或冒泡階段註冊事件偵聽器。這些階段在註冊期間是成對出現的,因爲冒泡階段只適用於目標節點的始祖。

如果不再需要某個事件偵聽器,可調用 removeEventListener() 刪除它,否則會產生內存問題。由於垃圾回收器不會刪除仍包含引用的對象,因此不會從內存中自動刪除使用已註冊事件偵聽器的對象。

複製 EventDispatcher 實例時並不複製其中附加的事件偵聽器。(如果新近創建的節點需要一個事件偵聽器,必須在創建該節點後附加該偵聽器。) 但是,如果移動 EventDispatcher 實例,則其中附加的事件偵聽器也會隨之移動。

如果在正在處理事件的節點上註冊事件偵聽器,則不會在當前階段觸發事件偵聽器,但會在事件流的稍後階段觸發,如冒泡階段。

如果從正在處理事件的節點中刪除事件偵聽器,則該事件偵聽器仍由當前操作觸發。刪除事件偵聽器後,決不會再次調用該事件偵聽器(除非再次註冊以備將來處理)。


參數  type:String — 事件的類型。 
 
listener:Function — 處理事件的偵聽器函數。此函數必須接受 Event 對象作爲其唯一的參數,並且不能返回任何結果,如下面的示例所示:
function(evt:Event):void函數可以有任何名稱。

 
useCapture:Boolean (default = false) — 確定偵聽器是運行於捕獲階段、目標階段還是冒泡階段。如果將 useCapture 設置爲 true,則偵聽器只在捕獲階段處理事件,而不在目標或冒泡階段處理事件。如果 useCapture 爲 false,則偵聽器只在目標或冒泡階段處理事件。要在所有三個階段都偵聽事件,請調用 addEventListener 兩次:一次將 useCapture 設置爲 true,一次將 useCapture 設置爲 false. 
 
priority:int (default = 0) — 事件偵聽器的優先級。優先級由一個帶符號的 32 位整數指定。數字越大,優先級越高。優先級爲 n 的所有偵聽器會在優先級爲 n -1 的偵聽器之前得到處理。如果兩個或更多個偵聽器共享相同的優先級,則按照它們的添加順序進行處理。默認優先級爲 0。 
 
useWeakReference:Boolean (default = false) — 確定對偵聽器的引用是強引用,還是弱引用。強引用(默認值)可防止您的偵聽器被當作垃圾回收。弱引用則沒有此作用。
類級別成員函數不屬於垃圾回收的對象,因此可以對類級別成員函數將 useWeakReference 設置爲 true 而不會使它們受垃圾回收的影響。如果對作爲嵌套內部函數的偵聽器將 useWeakReference 設置爲 true,則該函數將作爲垃圾回收並且不再是永久函數。如果創建對該內部函數的引用(將該函數保存到另一個變量中),則該函數將不作爲垃圾回收並仍將保持永久。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章