欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android 跨进程模拟按键(KeyEvent )实例详解

程序员文章站 2024-02-29 20:59:58
  android 解决不同进程发送keyevent 的问题 最近在做有关于remote controller 的功能,该功能把手机做成tv的遥控器来处理。在手...

  android 解决不同进程发送keyevent 的问题

最近在做有关于remote controller 的功能,该功能把手机做成tv的遥控器来处理。在手机的客户端发送消息到tv的android 服务端,服务端接收到客户端的请求消息,模拟keyevent命令,发送key值。

 最简单的发送命令为如下代码:

 
public static void simulatekeystroke(final int keycode) {
 

    new thread(new runnable() {
      
      public void run() {
        // todo auto-generated method stub
        try {
          
          instrumentation inst=new instrumentation();
          inst.sendkeydownupsync(keycode);
        } catch (exception e) {
          // todo: handle exception
        }
      }
    }).start();
  }

这种方法在当前的界面和相同的进程上是没有问题的,可以实现的基本的需求。但当我还是把服务开启着,按home将服务或者界面退出到后台时,再通过客户端向服务端服务发送消息使其模拟按键时,不幸的事情发送了:

injecting to another application requires inject_events permission

提示没有 inject_events这个权限。没则加之,在androidmanifest.xml文件里面添加该权限,再运行,问题还是没有解决,原因是上面代码最终还是调用的windowsmanagerservice 里面的injectkeyevent方法,该方法会去验证你当前的程序的pid和uid,如果两者在分发key 键时返回-1则会提示上面的error.

 好了,废话一大堆,下面到了真正解决这一问题的方法了。

网上各种google 各种百度,找不到自己需要的答案。

 想过一个方法是(尚未验证):

通过jni的方法将kernel 的发送keyevent的方法用ndk封装成方法,做成库给java调用,从而绕过android windowsmanagerservice 的验证,这是我初期想到的解决思路,但尚未验证。

另外一个通过验证的方法为:

将你的服务的userid改成系统级别的,在manifest加如下代码:

 <manifest xmlns:android="http://schemas.android.com/apk/res/android"

  package="com.xuzhitech.remote.server"
  android:versioncode="1"
  android:versionname="1.0"
  android:shareduserid="android.uid.system" >

 加上这一代码,需要在源码里面编译才能生效,添加android.mk文件:

local_path := $(call my-dir)
include $(clear_vars)

local_module_tags := optional

local_src_files := $(call all-subdir-java-files)

local_package_name := remoteandroidserver
local_certificate := platform
#local_certificate := share

local_overrides_packages := home
include $(build_package) 

这里的 local_certificate 要使用platform编译,而不是share编译。

到了这里,你就可以跨进程模拟按键了。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!