In one of my projects I need to track every screen touch event in background. That is, my app needs to be "invisible" while capturing every screen touch. Here is how I achieved this.
The idea is to define a dummy UI fragment that is really tiny (say, 1x1 pixel), and place it on one of the corners of the screen, and let it listen on all touch events outside it. Well, literally, it's not "invisible", in fact it's in foreground all the time! But since it's so tiny so hopefully users won't feel a difference.
First, let's create this dummy view:
mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
mDummyView = new LinearLayout(mContext);
LayoutParams params = new LayoutParams(1, LayoutParams.MATCH_PARENT);
mDummyView.setLayoutParams(params);
mDummyView.setOnTouchListener(this);
Here we set the width of the dummy view to be 1 pixel, and the height to be parent height. And we also set up a touch event listen of this dummy view, which we'll implement later.
Then let's add this dummy view.
LayoutParams params = new LayoutParams(
1, /* width */
1, /* height */
LayoutParams.TYPE_PHONE,
LayoutParams.FLAG_NOT_FOCUSABLE |
LayoutParams.FLAG_NOT_TOUCH_MODAL |
LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
PixelFormat.TRANSPARENT
);
params.gravity = Gravity.LEFT | Gravity.TOP;
mWindowManager.addView(mDummyView, params);
The key here is the FLAG_WATCH_OUTSIDE_TOUCH
flag, it enables the dummy view
to capture all events on screen, whether or not the event is inside the dummy
view or not.
Finally, let's handle the touch event by implementing View.OnTouchListener
listener.
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.d(TAG, "Touch event: " + event.toString());
// log it
return false;
}
We need to return false
since we're not really handling the event, so that the
underlying real UI elements can get those events.
A final note is that, to keep our dummy view always listening touch events, we
need to wrap all these in an Service
: we create the dummy view in onCreate
and add it to screen in onStartCommand
. And the service should implement
View.OnTouchListener
to receive the touch events.