Android webkit keyevent 事件傳遞過程

前言:基於android webview 上定製本身使用的可移植瀏覽器apk,遇到好多按鍵處理的問題。因此索性研究了一下keyevent 事件的傳遞流程。javascript

frameworks 層java

keyevent 事件開始是從/frameworks/base/core/java/android/webkit目錄下WebViewClassic.javanode

中的onKeyDown() 函數開始的android

          // Bubble up the key event if
          // 1. it is a system key; or
          // 2. the host application wants to handle it;
          if ((event.isSystem() || mCallbackProxy.uiOverrideKeyEvent(event))程序員

 這個的做用是判斷event是否是系統按鍵,或者調用webview應用處理event。系統按鍵直接返回,web

若是webview應用處理了也直接返回。瀏覽器

其它key事件調用 sendKeyEvent(event),在sendKeyEvent() 又調用sendBatchableInputMessage()微信

在這個函數中又調用mWebViewCore.sendMessage(message)app

將event封裝成Message傳遞給WebViewCore.java中的EventHub 類ide

在sendMessage()函數又經過它發送到Handler在transferMessages() 中handleMessage()處理keydown事件

          case KEY_DOWN:                                                                                                                 
                  key((KeyEvent) msg.obj, msg.arg1, true);
                  break;

webkit 層

key中調用nativeKey() 將事件傳入webkit中Source/WebKit/android/jni WebViewCore.cpp中的

    { "nativeKey", "(IIIIZZZZ)Z",                                                                                                                     
        (void*) Key },


WebViewCore::key(const PlatformKeyboardEvent& event)

eventHandler->keyEvent(event); 


此時調用進入Source/WebCore/page 中的EventHandler.cpp

它會區分爲keyup keydown keypress 事件發送到Node中處理

 bool Node::dispatchEvent(PassRefPtr<Event> event)
  {                
      return EventDispatcher::dispatchEvent(this, EventDispatchMediator(event));                                                   } 

經過中轉最終調用到EventDispatcher.cpp中

bool EventDispatcher::dispatchEvent(PassRefPtr<Event> event)

m_node->handleLocalEvents(event.get());

在Node.cpp 中調用

fireEventListeners(event); 

class Node : public EventTarget  Node繼承了EventTarget

EventTarget.cpp中實現註冊監聽

bool EventTarget::fireEventListeners(Event* event) 

 registeredListener.listener->handleEvent(scriptExecutionContext(), event); 

發送到註冊監聽的javascript中。


若是在js中註冊了一個keypress事件處理而咱們要兼容支持它咱們能夠只動WebViewClassic.java或者在app層代碼實現轉換並傳入js中便可。

在WebViewClassic.java中實現了passVirtualKeyEvent(int KeyCode)。

 

 若有問題和須要請留言,我會爲您解答問題。

 掃一掃下方二維碼或搜索微信號程序員互動聯盟(coder_online)便可關注,咱們能夠在線交流

相關文章
相關標籤/搜索