前言:基於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)便可關注,咱們能夠在線交流