手把手教你Android全局觸摸事件監聽
Android系統全局觸摸事件監聽
Android觸摸全局監聽指的是調用監聽後在任何界面都能獲取到觸摸事件。
要實現這個功能必須要修改源碼添加新的接口,因為系統默認是不暴露這個方法的。
源碼
監聽系統全局觸摸事件的類和相關代碼:
frameworks\base\services\core\java\com\android\server\wm\WindowManagerService.java @Override public void registerPointerEventListener(PointerEventListener listener, int displayId) { Slog.i(TAG, "registerPointerEventListener PointerEventListener = " + listener); synchronized (mGlobalLock) { final DisplayContent displayContent = mRoot.getDisplayContent(displayId); if (displayContent != null) { displayContent.registerPointerEventListener(listener); } } } @Override public void unregisterPointerEventListener(PointerEventListener listener, int displayId) { synchronized (mGlobalLock) { final DisplayContent displayContent = mRoot.getDisplayContent(displayId); if (displayContent != null) { displayContent.unregisterPointerEventListener(listener); } } }
第一個參數:是中PointerEventListener接口,
裡面有MotionEvent
對象含有點擊事件,比如DOWN、UP、MOVING
等其他信息。
package android.view; public interface WindowManagerPolicyConstants { interface PointerEventListener { void onPointerEvent(MotionEvent motionEvent); } }
第二個參數,屏幕id,正常用0 ,表示主屏幕id。有些設備有投屏或者第二屏才需要關註這個。
下面介紹如何註冊這個服務
1、綁定這個系統服務,這個方法行不通
因為這個服務的aidl接口IWindowManager,並沒有暴露這個方法
registerPointerEventListener方法定義在另一個內部接口 WindowManagerFuncs 中
public interface WindowManagerPolicy extends WindowManagerPolicyConstants { public interface WindowManagerFuncs { /** Register a system listener for touch events */ void registerPointerEventListener(PointerEventListener listener, int displayId); /** Unregister a system listener for touch events */ void unregisterPointerEventListener(PointerEventListener listener, int displayId); } }
2、獲取WindowManagerFuncs對象,該對象獲取的方式在源碼中有多種
參考:
frameworks\base\services\core\java\com\android\server\policy\PhoneWindowManager.java public PhoneWindowManager extends AbsPhoneWindowManager implements WindowManagerPolicy, IHwPhoneWindowManagerInner{ public WindowManagerFuncs getWindowManagerFuncs(){ return mWindowManagerFuncs; } }
WindowManagerFuncs在源碼中是可以直接new的,使用如下:
PhoneWindowManager phoneWindowManager = new PhoneWindowManager(); WindowManagerFuncs windowManagerFuncs = phoneWindowManager.getWindowManagerFuncs(); windowManagerFuncsEx.registerPointerEventListener(listener, Display.DEFAULT_DISPLAY);
3、在華為Emui源碼添加aidl回調
WindowManagerEx有通道直接發送數據到WindowManagerService並可以進行數據監聽
(1)添加aidl接口
vendor\huawei\Emui\frameworks\hwCommInterface\base\core\java\com\huawei\android\app\IHwPointEventCallback.aidl package com.huawei.android.app; import android.view.MotionEvent; oneway interface IHwPointEventCallback { void onPointerEvent(in MotionEvent motionEvent); }
(2)WindowManagerEx的修改
vendor\huawei\Emui\frameworks\hwext\hwext\framework\src\com\huawei\android\app\WindowManagerEx.java private final int TRANSACTION_SET_POINTER_EVENT_LISTENER = android.os.IBinder.FIRST_CALL_TRANSACTION + 2100; //給WindowManagerService傳遞監聽對象 public static void setPointerEventListener(IHwPointEventCallback listener) { Log.i(LOG_TAG, "setPointerEventListener listener = " + listener); IBinder windowManagerBinder = WindowManagerGlobal.getWindowManagerService().asBinder(); if (windowManagerBinder != null) { Parcel data = Parcel.obtain(); Parcel reply = Parcel.obtain(); try { data.writeInterfaceToken("android.view.IWindowManager"); //傳遞aidl監聽對象 data.writeStrongBinder(listener != null ? listener.asBinder() : null); //發送 windowManagerBinder.transact(TRANSACTION_SET_POINTER_EVENT_LISTENER, data, reply, 0); } catch (RemoteException e){ Log.e(LOG_TAG, "setPointerEventListener exception is " + e.getMessage()); } finally { data.recycle(); reply.recycle(); } } else { Log.w(LOG_TAG, "setPointerEventListener windowManagerBinder is null"); } }
(3)在WindowManagerService中接收數據並做實際監聽
基於盡量不修改源碼的理念,Emui中有WindowManagerService的子類HwWindowManagerService,在子類中修改代碼即可。
vendor\huawei\Emui\frameworks\base\services\java\huawei\com\android\server\wm\HwWindowManagerService.java HwWindowManagerService extends WindowManagerService private final int TRANSACTION_SET_POINTER_EVENT_LISTENER = android.os.IBinder.FIRST_CALL_TRANSACTION + 2100; private IHwPointEventCallback mIHwPointEventCallback = null; //接收WindowManagerEx傳遞過來的數據 public boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException { switch (code) { case TRANSACTION_SET_POINTER_EVENT_LISTENER: data.enforceInterface("android.view.IWindowManager"); IHwPointEventCallback observer = IHwPointEventCallback.Stub.asInterface(data.readStrongBinder()); setPointerEventListener(observer); reply.writeNoException(); return true; } } //在Service中創建唯一的監聽對象 private PointerEventListener mPointerEventListener = new PointerEventListener() { @Override public void onPointerEvent(MotionEvent motionEvent) { if(mIHwPointEventCallback != null) { try { mIHwPointEventCallback.onPointerEvent(motionEvent); } catch (RemoteException e) { Slog.e(TAG, "mIHwPointEventCallback error = " + e.getMessage()); } } } }; //添加設置觸摸監聽方法 private void setPointerEventListener(IHwPointEventCallback listener) { Slog.i(TAG, "setPointerEventListener PointerEventListener = " + listener); int uid = Binder.getCallingUid(); if(uid != Process.SYSTEM_UID){ Slog.e(TAG, "setPointerEventListener uid must be "+ Process.SYSTEM_UID +",but now uid = " + uid); return; } mIHwPointEventCallback = listener; if(listener != null) { //實際調到父類的註冊觸摸事件的方法 registerPointerEventListener(mPointerEventListener, Display.DEFAULT_DISPLAY); } else { //實際調到父類的反註冊觸摸事件的方法 unregisterPointerEventListener(mPointerEventListener, Display.DEFAULT_DISPLAY); } }
方法3可以實現在普通app中監聽到系統的全局觸摸事件,
因為app可以依賴Emui的emui_addons.jar,
調用到裡面的部分類,比如WindowManagerEx,就可以監聽全局觸摸事件。
其他系統環境可以根據實際情況參考上面的實現。
共勉:看得更多才知道還有更多還沒看過。
到此這篇關於手把手教你Android全局觸摸事件監聽的文章就介紹到這瞭,更多相關Android觸摸監聽內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 華為鴻蒙系統應用開發工具 DevEco Studio的安裝和使用圖文教程
- 解析Android AIDL的實例與原理
- Android音視頻開發Media FrameWork框架源碼解析
- Android點擊事件之多點觸摸與手勢識別的實現
- Android中EditText光標的顯示與隱藏方法