一、APP通过View修改鼠标样式

  app view上修改鼠标样式比较简单,通过 hover event 获取鼠标坐标并使用如下方法修改为自定义图片:

  getWindow().getDecorView().setPointerIcon(PointerIcon.load(getResources(), R.drawable.pointer_spot_touch_icon));

        imageView = (ImageView) findViewById(R.id.image_view);
imageView.setOnHoverListener(new View.OnHoverListener() {
@SuppressLint({"SetTextI18n", "ResourceType"})
@Override
public boolean onHover(View v, MotionEvent event) {
int what = event.getAction(); textX.setText("X : " + event.getX());
textY.setText("Y : " + event.getY()); switch(what){
case MotionEvent.ACTION_HOVER_ENTER: //鼠标进入view
Log.i(TAG, "bottom ACTION_HOVER_ENTER...");
mOrgPI = getWindow().getDecorView().getPointerIcon();
getWindow().getDecorView().setPointerIcon(PointerIcon.load(getResources(), R.drawable.pointer_spot_touch_icon));
break;
case MotionEvent.ACTION_HOVER_MOVE: //鼠标在view上
Log.i(TAG, "bottom ACTION_HOVER_MOVE...");
break;
case MotionEvent.ACTION_HOVER_EXIT: //鼠标离开view
Log.i(TAG, "bottom ACTION_HOVER_EXIT...");
getWindow().getDecorView().setPointerIcon(mOrgPI);
break;
}
return false;
}
});
}

 其中pointer_spot_touch_icon.xml 需要声明为 pointer-icon :

<?xml version="1.0" encoding="utf-8"?>
<pointer-icon xmlns:android="http://schemas.android.com/apk/res/android"
android:bitmap="@drawable/pointer_red_dot_arrow"
android:hotSpotX="6dp"
android:hotSpotY="6dp" />

 但是app修改鼠标样式的view关闭后,鼠标样式会恢复成默认的黑箭头,因此不依赖APP去动态切换鼠标样式需要在framework层修改系统源码实现。

二、framework层添加自定义鼠标样式并通过按键切换

(1)添加自定义样式资源

  系统图标资源在 frameworks/base/core/res/res/drawable-mdpi/ 目录,其中 pointer_arrow.png、pointer_arrow_large.png 是系统默认的黑色箭头,

  pointer_arrow_red_dot.png、pointer_arrow_red_dot_large.png 是自己添加的红点样式图片:

 然后在 frameworks/base/core/res/res/drawable/ 目录添加对应的xml:

 pointer_arrow_red_dot_icon.xml

<?xml version="1.0" encoding="utf-8"?>
<pointer-icon xmlns:android="http://schemas.android.com/apk/res/android"
android:bitmap="@drawable/pointer_arrow_red_dot"
android:hotSpotX="5dp"
android:hotSpotY="5dp" />

  pointer_arrow_red_dot_large_icon.xml

<?xml version="1.0" encoding="utf-8"?>
<pointer-icon xmlns:android="http://schemas.android.com/apk/res/android"
android:bitmap="@drawable/pointer_arrow_red_dot_large"
android:hotSpotX="10dp"
android:hotSpotY="10dp" />

修改 frameworks/base/core/res/res/values/styles.xml 添加资源配置,注意名字的匹配!

修改 frameworks/base/core/res/res/values/attrs.xml  引用资源:

(2)Java 层获取资源

  修改 frameworks/base/core/java/android/view/PointerIcon.java ,添加如下定义:

  在 getSystemIconTypeIndex(int type) 函数中返回之前配置的资源:

(3)c++层添加对应的id并加载资源

 修改 frameworks/base/core/jni/android_view_PointerIcon.h

* Pointer icon styles.
* Must match the definition in android.view.PointerIcon.
*/
enum {
POINTER_ICON_STYLE_CUSTOM = -,
POINTER_ICON_STYLE_NULL = ,
POINTER_ICON_STYLE_ARROW = ,
POINTER_ICON_STYLE_CONTEXT_MENU = ,
POINTER_ICON_STYLE_HAND = ,
POINTER_ICON_STYLE_HELP = ,
POINTER_ICON_STYLE_WAIT = ,
POINTER_ICON_STYLE_CELL = ,
POINTER_ICON_STYLE_CROSSHAIR = ,
POINTER_ICON_STYLE_TEXT = ,
POINTER_ICON_STYLE_VERTICAL_TEXT = ,
POINTER_ICON_STYLE_ALIAS = ,
POINTER_ICON_STYLE_COPY = ,
POINTER_ICON_STYLE_NO_DROP = ,
POINTER_ICON_STYLE_ALL_SCROLL = ,
POINTER_ICON_STYLE_HORIZONTAL_DOUBLE_ARROW = ,
POINTER_ICON_STYLE_VERTICAL_DOUBLE_ARROW = ,
POINTER_ICON_STYLE_TOP_RIGHT_DOUBLE_ARROW = ,
POINTER_ICON_STYLE_TOP_LEFT_DOUBLE_ARROW = ,
POINTER_ICON_STYLE_ZOOM_IN = ,
POINTER_ICON_STYLE_ZOOM_OUT = ,
POINTER_ICON_STYLE_GRAB = ,
POINTER_ICON_STYLE_GRABBING = , POINTER_ICON_STYLE_SPOT_HOVER = ,
POINTER_ICON_STYLE_SPOT_TOUCH = ,
POINTER_ICON_STYLE_SPOT_ANCHOR = , POINTER_ICON_STYLE_REDDOT = , //增加自定义样式的枚举定义,与上面 PointerIcon.java 中的变量对应
};

 修改 frameworks/base/services/core/jni/com_android_server_input_InputManagerService.cpp ,加载到自定义枚举变量对应的图片资源:

void NativeInputManager::loadAdditionalMouseResources(std::map<int32_t, SpriteIcon>* outResources,
std::map<int32_t, PointerAnimation>* outAnimationResources) {
JNIEnv* env = jniEnv(); for (int iconId = POINTER_ICON_STYLE_CONTEXT_MENU; iconId <= POINTER_ICON_STYLE_REDDOT;
++iconId) {
PointerIcon pointerIcon;
loadSystemIconAsSpriteWithPointerIcon(
env, mContextObj, iconId, &pointerIcon, &((*outResources)[iconId]));
if (!pointerIcon.bitmapFrames.empty()) {
PointerAnimation& animationData = (*outAnimationResources)[iconId];
size_t numFrames = pointerIcon.bitmapFrames.size() + ;
animationData.durationPerFrame =
milliseconds_to_nanoseconds(pointerIcon.durationPerFrame);
animationData.animationFrames.reserve(numFrames);
animationData.animationFrames.push_back(SpriteIcon(
pointerIcon.bitmap, pointerIcon.hotSpotX, pointerIcon.hotSpotY));
for (size_t i = ; i < numFrames - ; ++i) {
animationData.animationFrames.push_back(SpriteIcon(
pointerIcon.bitmapFrames[i], pointerIcon.hotSpotX, pointerIcon.hotSpotY));
}
}
}
loadSystemIconAsSprite(env, mContextObj, POINTER_ICON_STYLE_NULL,
&((*outResources)[POINTER_ICON_STYLE_NULL]));
}

(4)按键切换鼠标样式

  按键事件处理在 frameworks/base/services/core/java/com/android/server/policy/PhoneWindowManager.java 的 interceptKeyBeforeDispatching 函数中

......
String keytype = event.getDevice().getName();
int vendorId = event.getDevice().getVendorId();
int productId = event.getDevice().getProductId();
Log.d(TAG, "onKeyDown send keycode : " + keyCode +
", keytype : " + keytype +
", vendorId : " + vendorId +
", productId : " + productId); if (event.getAction() == KeyEvent.ACTION_UP &&
vendorId == 6421 && productId == 4146) { //过滤指定VID PID 的HID设备按键值 /* @id:
*TYPE_NULL = 0;
*TYPE_ARROW = 1000;
*TYPE_OEM_FIRST = 10000;
*TYPE_ARROW_REDDOT = 10001;
*/ switch (keyCode) { case 139: // air mouse
InputManager.getInstance().setPointerIconType(PointerIcon.TYPE_ARROW_REDDOT); //设置指定id的鼠标样式
SystemProperties.set("persist.sys.lxl.mouse_id", "10001"); //通过属性记录当前样式ID,目的是防止鼠标样式被其他界面更新,后面会介绍到
break; default:
break;
}
}
......

 其中 setPointerIconType() 方法实现在  frameworks/base/services/core/java/com/android/server/input/InputManagerService.java

    // Binder call
@Override
public void setPointerIconType(int iconId) {
nativeSetPointerIconType(mPtr, iconId);
}

 接着调用到 native层的方法,实现在 frameworks/base/services/core/jni/com_android_server_input_InputManagerService.cpp

static void nativeSetPointerIconType(JNIEnv* /* env */, jclass /* clazz */, jlong ptr, jint iconId) {
NativeInputManager* im = reinterpret_cast<NativeInputManager*>(ptr);
im->setPointerIconType(iconId);
}
void NativeInputManager::setPointerIconType(int32_t iconId) {
AutoMutex _l(mLock);
sp<PointerController> controller = mLocked.pointerController.promote();
if (controller != NULL) {
controller->updatePointerIcon(iconId);
}
}

 updatePointerIcon()方法实现在 frameworks/base/libs/input/PointerController.cpp 中:

void PointerController::updatePointerIcon(int32_t iconId) {
AutoMutex _l(mLock); //lxl add for custom mouse icon start
const int32_t customIconId = mPolicy->getCustomPointerIconId();
if(customIconId >= ) {
iconId = customIconId;
}
//lxl add for custom mouse icon end if (mLocked.requestedPointerType != iconId) {
mLocked.requestedPointerType = iconId;
mLocked.presentationChanged = true;
updatePointerLocked();
}
}

 其中就是通过 getCustomPointerIconId() 去获取当前属性id,强制更新为自定义的样式:

int32_t NativeInputManager::getCustomPointerIconId() {

     //lxl add for custom mouse icon start
int mVaule;
char mgetVal[PROPERTY_VALUE_MAX+]={}; property_get("persist.sys.lxl.mouse_id",mgetVal,""); //例如鼠标悬浮在按钮上时会变成小手样式,需要使用到前面按键时设定的属性值,强制更新为自定义鼠标样式。
mVaule = atoi(mgetVal);
switch (mVaule)
{
case POINTER_ICON_STYLE_REDDOT:
case POINTER_ICON_STYLE_NULL:
return mVaule;
default:
break;
}
//lxl add for custom mouse icon end
return POINTER_ICON_STYLE_CUSTOM;
}

 (5)隐藏鼠标样式接口

   同样在frameworks/base/services/core/jni/com_android_server_input_InputManagerService.cpp 中添加一个JNI方法:

//lxl add for custom mouse icon start
static void android_server_InputManager_nativefadeMouse(JNIEnv* env,
jclass clazz,jlong ptr){
NativeInputManager* im = reinterpret_cast<NativeInputManager*>(ptr); int mID;
char *mgetID=new char[PROPERTY_VALUE_MAX]; property_get("sys.ID.mID",mgetID,);
mID=atoi(mgetID); static sp<PointerControllerInterface>mPointerController=im->obtainPointerController(mID); ALOGI("Fade mouse by user"); //start to dispatchMouse
mPointerController->setPresentation(
PointerControllerInterface::PRESENTATION_POINTER);
mPointerController->fade(PointerControllerInterface::TRANSITION_IMMEDIATE);
}
//lxl add for custom mouse icon end

  并添加到 JNINativeMethod gInputManagerMethods[] 数组中去:

//lxl add for custom mouse icon start
{ "nativefadeMouse", "(J)V",
(void*) android_server_InputManager_nativefadeMouse},
//lxl add for custom mouse icon end

  然后在 frameworks/base/services/core/java/com/android/server/input/InputManagerService.java 即可声明使用:

//lxl add for custom mouse icon start
private static native void nativefadeMouse(long ptr);
public void fadeMouse(){
  nativefadeMouse(mPtr);
}
//lxl add for custom mouse icon end

  例如在 systemRunning() 中注册一个广播监听 USB 设备的插拔,在指定VID PID的HID设备拔除时调用fade接口隐藏鼠标:

        //lxl add for custom mouse icon start
filter = new IntentFilter();
filter.addAction(UsbManager.ACTION_USB_DEVICE_ATTACHED);
filter.addAction(UsbManager.ACTION_USB_DEVICE_DETACHED); mContext.registerReceiver(new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
Bundle bundle = intent.getExtras();
String action = intent.getAction();
String idStr = bundle.getString("id");
Slog.i(TAG, "onReceive : " + idStr); if (action.equals(UsbManager.ACTION_USB_DEVICE_ATTACHED)) { // USB设备接入
Log.d(TAG, "onReceive: " + "ACTION_USB_DEVICE_ATTACHED"); UsbDevice device = (UsbDevice) intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
if (null == device) {
Log.w(TAG, "null usb device");
return;
} Log.d(TAG, "--->getDeviceName:"+device.getDeviceName() +
",getVendorId:"+device.getVendorId() +
",getProductId:"+device.getProductId()); int count = device.getConfigurationCount();
for (int i = 0; i < count; i++) {
UsbConfiguration configuration = device.getConfiguration(i);
if (null == configuration) {
Log.w(TAG, "null usb configuration");
return;
}
int interfaceCount = configuration.getInterfaceCount();
for (int j = 0; j < interfaceCount; j++) {
UsbInterface usbInterface = configuration.getInterface(j);
if (null != usbInterface && usbInterface.getInterfaceClass() == UsbConstants.USB_CLASS_HID) { Log.d(TAG, "onReceive: " + "A hid device connected, mId: "+usbInterface.getId()
+ ", mName: " + usbInterface.getName());
}
}
} } else if (action.equals(UsbManager.ACTION_USB_DEVICE_DETACHED)) { // USB设备移除
Log.d(TAG, "onReceive: " + "ACTION_USB_DEVICE_DETACHED"); UsbDevice device = (UsbDevice) intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
if (null == device) {
Log.w(TAG, "null usb device");
return;
} Log.d(TAG, "--->getDeviceName:"+device.getDeviceName() +
",getVendorId:"+device.getVendorId() +
",getProductId:"+device.getProductId()); int count = device.getConfigurationCount();
for (int i = 0; i < count; i++) {
UsbConfiguration configuration = device.getConfiguration(i);
if (null == configuration) {
Log.w(TAG, "null usb configuration");
return;
}
int interfaceCount = configuration.getInterfaceCount();
for (int j = 0; j < interfaceCount; j++) {
UsbInterface usbInterface = configuration.getInterface(j);
if (null != usbInterface && usbInterface.getInterfaceClass() == UsbConstants.USB_CLASS_HID) { Log.d(TAG, "onReceive: " + "A hid device disconnected, mId: "+usbInterface.getId()
+ ", mName: " + usbInterface.getName()); if(device.getVendorId() == 6421 && device.getProductId() == 4146){
fadeMouse(); // 隐藏鼠标
}
}
}
}
}
}
}, filter, null, mHandler);
//lxl add for custom mouse icon end

(6)HID设备数据监听

  实际项目中,需要监测HID设备数据上报状态,所以在 frameworks/native/services/inputflinger/EventHub.cpp 中添加了一个线程+定时器进行实时监测,超时则认为设备移除,恢复默认鼠标样式。

     定时器中通过am发送广播给framework层更新鼠标样式(可参考InputManagerService.java 中 USB 设备插拔广播添加),定时器方法如下:

//lxl add for custom mouse icon start
static pthread_t mTimerThread = -;
static int mCountDown = ;
static bool mThreadRun = false;

// 定时器处理函数
static void handle(union sigval v)
{
time_t t;
char p[]; time(&t);
strftime(p, sizeof(p), "%T", localtime(&t)); ALOGV("%s thread %lu, val = %d, mCountDown = %d , mThreadRun = %d\n",
p, pthread_self(), v.sival_int, mCountDown, mThreadRun); mCountDown--;
if(mCountDown <= ){
mCountDown = ;
mThreadRun = false;
} return;
}

// 线程启动函数
static void* cursorTimer(void* p)
{
struct sigevent evp;
struct itimerspec ts;
timer_t timer;
String8 sys_cmd;
int ret; memset (&evp, , sizeof (evp));
evp.sigev_value.sival_ptr = &timer;
evp.sigev_notify = SIGEV_THREAD;
evp.sigev_notify_function = handle;
evp.sigev_value.sival_int = ; // handle() args ret = timer_create(CLOCK_REALTIME, &evp, &timer); // 创建定时器
if(ret)
ALOGV("cursorTimer Create");// send broadcast to update icon when get air mouse data
sys_cmd = "am broadcast -a com.lxl.action.mouseicon --es id \"ENABLE\" -f 0x01000000";
system(sys_cmd); ts.it_interval.tv_sec = ;
ts.it_interval.tv_nsec = ;
ts.it_value.tv_sec = ;
ts.it_value.tv_nsec = ; ret = timer_settime(timer, TIMER_ABSTIME, &ts, NULL);
if(ret)
ALOGV("timer_settime"); while(mThreadRun){
usleep();
ALOGV("cursorTimer is running...");
} ret = timer_delete (timer);
if(!ret)
ALOGV("cursorTimer Delete");// send broadcast to update icon when get air mouse data
sys_cmd = "am broadcast -a com.lxl.action.mouseicon --es id \"DISABLE\" -f 0x01000000";
system(sys_cmd); mTimerThread = -; ALOGV("cursorTimer Exit");
return NULL;
}
//lxl add for custom mouse icon end

 在EventHub::getEvents() 方法中创建该线程:

         ......
     //lxl add for custom mouse icon start
if(device->identifier.vendor == 0x1915 && device->identifier.product == 0x1032) {
if(iev.type == && (iev.code == || iev.code == )) { // cursor data mCountDown = 30; // 有数据上传则重置定时器倒计时30s
// creat timer thread for sending broadcast to reset icon
if(mTimerThread < ) {
mThreadRun = true;
int ret = pthread_create(&mTimerThread, NULL, &cursorTimer, NULL); // 创建定时器线程
if(!ret)
ALOGV("Create air mouse timer thread successed");
}
} else if(iev.type== && iev.code== && iev.value == ) { // air mouse key down if(mCountDown > ) { // air mouse is enable, then disable right now
mThreadRun = false; // Let timer thread exit
mCountDown = ;
}
}
}
//lxl add for custom mouse icon end
    ......

  以上实现方式和交互逻辑可根据实际项目需求合理设计~

Android:系统自定义鼠标样式切换的更多相关文章

  1. CSharp如何自定义鼠标样式

    一.如何设置鼠标样式? 在CSharp的WinForm开发中,可以通过下面的API设置鼠标样式: //把鼠标样式设置为十字(系统自带的一种鼠标样式) this.Cursor = Cursors.Cro ...

  2. .Net应用自定义鼠标样式

    (调用系统API的方法) 1.引用命名空间 using System.Runtime.InteropServices; 命名空间提供各种各样支持 COM 互操作 及平台调用服务的成员.using Sy ...

  3. Android中实现全屏、无标题栏的两种办法(另附Android系统自带样式的解释)

    在进行UI设计时,我们经常需要将屏幕设置成无标题栏或者全屏.要实现起来也非常简单,主要有两种方法:配置xml文件和编写代码设置. 1.在xml文件中进行配置 在项目的清单文件AndroidManife ...

  4. C#、WPF中如何自定义鼠标样式

    需求:在C#中如何自定义鼠标样式?在这里可以分两种情况,一种是在winForm,另一种是在WPF中(注意使用的Cursor对象不一样) 解决办法如下: a.首先针对WinForm中,我们可以采用图标加 ...

  5. Arcgis for javascript不同的状态下自定义鼠标样式

    俗话说:爱美之心,人皆有之.是的,没错,即使我只是一个做地图的,我也希望自己的地图看起来好看一点.在本文,给大家讲讲在Arcgis for javascript下如何自定义鼠标样式. 首先,说几个状态 ...

  6. [转]Android中自定义checkbox样式

    android中自定义checkbox的图片和大小   其实很简单,分三步: 1.在drawable中创建文件checkbox_selector.xml: <?xml version=" ...

  7. ArcGIS AddIN开发之自定义鼠标样式

    如果想修改Windows默认的鼠标样式,可以这样 //设置鼠标样式 this.Cursor = System.Windows.Forms.Cursors.Cross; 可是如果想设置成一些自定义的很好 ...

  8. cursor url 自定义鼠标样式

    cursor可以自定义鼠标,写法是cursor:url(“图片路径”),pointer; url:需使用的自定义光标的 URL.图片类型需要是.cur或.ani和jpg,png等格式的(.cur或.a ...

  9. 更改win系统的鼠标样式

    一.找一个你心仪的鼠标样式(.cur文件),并放到 C:\Windows\Cursors 目录下 二.打开,控制面板 -> 硬件和声音 -> 鼠标 ,如下图: 三.浏览鼠标目录,找到你存放 ...

随机推荐

  1. CentOS7.X版本系统的下载和安装

    一.下载CentOS镜像 1.打开浏览器输入centos.org 2.选择版本下载 3.进入下载页面选择下载版本的种子链接,在迅雷下载即可. 二安装CentOS系统 1.服务器开机,根据界面提示进入磁 ...

  2. sqlalchemy.exc.CompileError: (in table 'user', column 'username'): VARCHAR requires a length on dialect mysql

    映射数据库时报错:sqlalchemy.exc.CompileError: (in table 'user', column 'username'): VARCHAR requires a lengt ...

  3. Tensorflow&CNN:裂纹分类

    版权声明:本文为博主原创文章,转载 请注明出处:https://blog.csdn.net/sc2079/article/details/90478551 - 写在前面 本科毕业设计终于告一段落了.特 ...

  4. JVM系列二:垃圾回收

    什么时候回收对象 引用计数法 1.原理:为对象添加一个引用计数器,当对象增加一个引用时计数器加 1,引用失效时计数器减 1.引用计数为 0 的对象可被回收. 2.缺点:无法解决循环引用问题 可达性分析 ...

  5. 【原创】SPFA判负环

    [定义与概念] 给定一张有向图,若其中存在一个环的所有权值之和为负数,这个环称为负环. [算法实现] 当然,负环的求解可以暴搜,但是时间复杂度就难以入眼了,我们回到求解单源最短路径算法上面,看看它们能 ...

  6. JS创建SVG的问题

    在线编辑的一个东西,用的是js+svg,遇到了这样一个问题,就是说我监听页面的单击事件,然后记录下来鼠标单击的位置,给svg添加子标签,然后页面上展示出来 说的可能不大清楚,上代码吧 <!DOC ...

  7. C++——宏观把控

    跟看所有的书一样,我们都要求第一遍泛读,宏观把控书本内容,C++依旧如此进行.看到前面这几章的时候感觉非常熟悉,因为能让我联想到很多以前学习的VB.C#等的知识,感觉轻松很多,原来我已经学过了很多东西 ...

  8. git 将master分支合到自己的开发分支

    背景: 一般开发自己的分支都是从最新的master上拉取,但中间master会有改动,此时需要将最新的master合到自己的分支中 命令: 1. 查看当前的分支,星号标识为当前分支:(如果查询结果有m ...

  9. ARRAY_MAP函数用法

    ARRAY_MAP函数用法 Posted on 2012-9-18, 22:15, by tmser, under php 总结 . 看php JSON 类中有这样一种用法array_map(arra ...

  10. Vue Router 使用方法

    安装 直接下载 / CDN https://unpkg.com/vue-router/dist/vue-router.js Unpkg.com 提供了基于 NPM 的 CDN 链接.上面的链接会一直指 ...