从技术角度出发,就是H5上的“Pointer Lock”在Android浏览器上无法支持到。
https://www.chromestatus.com/feature/6753200417800192
Pointer Lock的效果是:
- 隐藏鼠标
- 给原生的鼠标事件到对应的节点,即传输鼠标移动的相对坐标
有哪些解决办法
解决办法一:
可以用 requestPointerCapture 来实现这个功能。
通过时间注入,可以在部分系统(Android8.0)上彻底解决问题。
对应的方法参考这个文档:
https://developer.android.com/training/gestures/movement#pointer-capture
具体来说:
可以通过升级系统至Android8.0+原生支持PointerCapture 可以完全实现鼠标沉浸式体验。
public class MainActivity extends AppCompatActivity {
private View mRootView = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mRootView = findViewById(R.id.root);
mRootView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
v.requestPointerCapture();
}
});
mRootView.setOnCapturedPointerListener(new View.OnCapturedPointerListener() {
@Override
public boolean onCapturedPointer(View view, MotionEvent motionEvent) {
Log.e("Yeshen", motionEvent.getAction() + ",x:" + motionEvent.getX() + ",y:" + motionEvent.getY());
return true;
}
});
}
@Override
public void onPause() {
super.onPause();
mRootView.releasePointerCapture();
}
}
解决办法二:
考虑到解决办法一需要升级系统,开发成本不低,方法二是想找些便宜但是不彻底的方法。
期望是不隐藏鼠标,但是要读取到鼠标的相对位移。
当鼠标移动时,首次触发的是硬件驱动,驱动收到事件后,将该相应事件写入到输入设备节点, 这便产生了最原生态的内核事件。接着,输入系统取出原生态的事件,经过层层封装后成为KeyEvent或者MotionEvent ;最后,交付给相应的目标窗口(Window)来消费该输入事件。
鼠标绘制的核心的代码在这里PointerController.cpp。这个边界也应该是在这个类中处理的。从166行的代码可以猜测,就算在上层获得鼠标位移事件,也无法绕开这个边界。
我们期望收到的Mouse move或者说hover事件,应该是从这个类派生出来的。
尝试一:
Activity 中重写 onGenericMotionEvent 可以获得位移。
自测后发现,屏幕是有边界的,到达边界之后,位移就消失了。
PS:有没有可能从 USB driver/InputDispatcher 中直接获取位移数据呢?
// TODO
小结
Android8.0+ 可以彻底修复问题。
其他资料:
- android7.0 支持自定义鼠标样式:https://developer.android.com/about/versions/nougat/android-7.0#custom_pointer_api
- Android Overlay Mouse Cursor:https://www.pocketmagic.net/android-overlay-cursor/
- USB事件监听:https://developer.android.com/guide/topics/connectivity/usb/host