Android 媒体键监听以及模拟媒体键盘的实现 demo
有时我们需要程序模拟按钮或点击,而手机本身又没有,哪么可以采取其它方式 模拟实现,最后再去实际设备去测试(前期一般都拿不到设备);
如上一首,下一首,暂停等,手机上是没有的,但有些设备上是有的,所以我们只能模拟;
模拟按钮一种可以采用adb 命令;
别一种采用程序,这里主要讲采用程序的方法:
通过Runtime实现,模拟媒体上一首代码如下:
try
{
String keyCommand = "input keyevent " + KeyEvent.KEYCODE_MEDIA_NEXT;
Runtime runtime = Runtime.getRuntime();
Process proc = runtime.exec(keyCommand); }
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
模拟上一首
try
{
String keyCommand = "input keyevent " + KeyEvent.KEYCODE_MEDIA_PREVIOUS;
Runtime runtime = Runtime.getRuntime();
Process proc = runtime.exec(keyCommand);
}
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
如果需要模拟其它按键只需将KeyEvent.KEYCODE_MEDIA_PREVIOUS替换成其它键值即可。
下面讲一下 媒体键监听:
首先我们定义一个广播类 MediaButtonReceive 它继承广播接收器类(BroadbcastReceiver),那么它就具备了BroadbcastReceiver类的使用方式,
点击MEDIA_BUTTON发送的Intent Action 为:ACTION_MEDIA_BUTTON ="android.intent.action.MEDIA_BUTTON"
Intent 附加值为(Extra)点击MEDIA_BUTTON的按键码 :
//获得KeyEvent对象
KeyEvent keyEvent = (KeyEvent)intent.getParcelableExtra(Intent.EXTRA_KEY_EVENT);
//获得Action
String intentAction = intent.getAction() ;
在取得音频焦点时,在AudioManager对象注册一个MediaoButtonRecevie,使它成为MEDIA_BUTTON的唯一接收器 也就是说只有它能收到,其他的都收不到这个广播了,否则的话会造成的混乱,在失去音频焦点时反注册,这样才能保证其它应用能正常使用媒体键;
该广播必须在AndroidManifest.xml文件中进行声明,否则就监听不到该MEDIA_BUTTON广播了
在AudioManager对象注册一个MediaoButtonRecevie;
下面看一下MediaButtonReceiver类 的实现:
import android.content.Context;
import android.content.Intent;
import android.util.Log;
import android.view.KeyEvent;
import android.widget.Toast; public class MediaButtonReceiver extends BroadcastReceiver { @Override
public void onReceive(Context context, Intent intent) {
// 获得Action
String intentAction = intent.getAction(); // 获得KeyEvent对象
KeyEvent keyEvent = (KeyEvent) intent
.getParcelableExtra(Intent.EXTRA_KEY_EVENT); Log.e("MediaButtonReceiver", "Action ---->" + intentAction
+ " KeyEvent----->" + keyEvent.toString()); if (Intent.ACTION_MEDIA_BUTTON.equals(intentAction)) { boolean isActionUp = (keyEvent.getAction()==KeyEvent.ACTION_UP);
// 这里会收到两次,我们只判断 up
if(!isActionUp)
{
return;
} // 获得按键字节码
int keyCode = keyEvent.getKeyCode();
// 按下 / 松开 按钮
int keyAction = keyEvent.getAction();
// 获得事件的时间
long downtime = keyEvent.getEventTime(); // 获取按键码 keyCode
StringBuilder sb = new StringBuilder();
// 这些都是可能的按键码 , 打印出来用户按下的键
if (KeyEvent.KEYCODE_MEDIA_NEXT == keyCode) {
sb.append("KEYCODE_MEDIA_NEXT");
Toast.makeText(context,
"receive KEYCODE_MEDIA_NEXT",
Toast.LENGTH_SHORT).show();
} if (KeyEvent.KEYCODE_MEDIA_PREVIOUS == keyCode) {
sb.append("KEYCODE_MEDIA_PREVIOUS");
Toast.makeText(context,
"receive KEYCODE_MEDIA_PREVIOUS",
Toast.LENGTH_SHORT).show();
} // 输出点击的按键码
Log.e("MediaButtonReceiver", sb.toString());
}
} }
在AndroidManifest.xml声明我们定义的广播类,它需要通过AudioManager对象注册
<span style="font-size:18px;"> <receiver android:name="MediaButtonReceiver">
<intent-filter >
<action android:name="android.intent.action.MEDIA_BUTTON"></action>
</intent-filter>
</receiver></span>
AudioManager注册一个MediaButtonReceiver() ;
<span style="font-size:18px;"> //获得AudioManager对象
AudioManager mAudioManager =(AudioManager)getSystemService(Context.AUDIO_SERVICE);
//构造一个ComponentName,指向MediaoButtonReceiver类
//下面为了叙述方便,我直接使用ComponentName类来替代MediaoButtonReceiver类
ComponentName mbCN = new ComponentName(getPackageName(),MediaButtonReceiver.class.getName());
//注册一个MedioButtonReceiver广播监听
mAudioManager.registerMediaButtonEventReceiver(mbCN);</span>
下面看一下调用类的实现MainActivity .java
:
import com.example.ydsdkdemo.R; import android.media.AudioManager;
import android.os.Bundle;
import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.util.Log;
import android.view.KeyEvent;
import android.view.Menu;
import android.view.View;
import android.widget.Toast; public class MainActivity extends Activity { private Context mContext;
private AudioManager audioManager;
private ComponentName mComponentName; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mContext = this;
init(); } private void init() { audioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);
mComponentName = new ComponentName(getPackageName(),
MediaButtonReceiver.class.getName()); if (AudioManager.AUDIOFOCUS_REQUEST_GRANTED == audioManager
.requestAudioFocus(focusChangeListener,
AudioManager.STREAM_MUSIC, AudioManager.AUDIOFOCUS_GAIN)) {
audioManager.registerMediaButtonEventReceiver(mComponentName);
} try
{
String keyCommand = "input keyevent " + KeyEvent.KEYCODE_MEDIA_NEXT;
Runtime runtime = Runtime.getRuntime();
Process proc = runtime.exec(keyCommand); }
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
} } public void lastOne(View v)
{
try
{
String keyCommand = "input keyevent " + KeyEvent.KEYCODE_MEDIA_PREVIOUS;
Runtime runtime = Runtime.getRuntime();
Process proc = runtime.exec(keyCommand);
}
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
} public void nextOne(View v)
{
try
{
String keyCommand = "input keyevent " + KeyEvent.KEYCODE_MEDIA_NEXT;
Runtime runtime = Runtime.getRuntime();
Process proc = runtime.exec(keyCommand); }
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
} public void voiceSearch(View v)
{
try
{
String keyCommand = "input keyevent " + KeyEvent.KEYCODE_MEDIA_RECORD;
Runtime runtime = Runtime.getRuntime();
Process proc = runtime.exec(keyCommand);
}
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
} //焦点问题
private AudioManager.OnAudioFocusChangeListener focusChangeListener = new AudioManager.OnAudioFocusChangeListener() {
@Override
public void onAudioFocusChange(int focusChange) {
switch (focusChange) { case AudioManager.AUDIOFOCUS_LOSS:// 长时间失去 Toast.makeText(mContext,
"receive AudioManager.AUDIOFOCUS_LOSS",
Toast.LENGTH_SHORT).show();
audioManager.unregisterMediaButtonEventReceiver(mComponentName);
audioManager.abandonAudioFocus(focusChangeListener);//放弃焦点监听
break; case AudioManager.AUDIOFOCUS_LOSS_TRANSIENT:// 短时间失去
case AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK:// 短时间失去,但可以共用 break; case AudioManager.AUDIOFOCUS_GAIN:// 获得音频焦点
audioManager.registerMediaButtonEventReceiver(mComponentName);
break; }
}
}; @Override
protected void onDestroy() {
// TODO Auto-generated method stub
audioManager.unregisterMediaButtonEventReceiver(mComponentName);
super.onDestroy();
} }
值得注意的一点时,当我们为一个应用程序注册了MediaoButtonReceiver时,在程序离开时,我们需要取消该
Demo地址:
http://download.csdn.net/detail/q610098308/9147909
Android 媒体键监听以及模拟媒体键盘的实现 demo的更多相关文章
- Android Home键监听
Android Back Home键监听 Back键的监听 对于Back键的监听比较容易,可以在多个系统回调处拦截,比如在activity的下列方法中都可以收到Back键按下的事件: @Overrid ...
- quick-cocos2d-x android返回键监听并实现原生退出对话框
这两天最终闲了一下,就顺手又把quick捡起来又学了学,一直都认为quick比cocos2dx那套lua绑定要方便很多,今天试了下android返回键的监听,还是挺好弄的,所以就有了这篇. 首先说明一 ...
- 使用python监听、模拟鼠标键盘事件
最近守望职业选手疑似开挂事件挺热闹的,在下小菜一枚,并不能从视频中看出端倪.看了一些关于外挂的讨论,自动点射和压枪只需在鼠标驱动上做些改动即可,自瞄或其他高级功能则需要读内存或修改游戏文件,检测也更容 ...
- Android Back Home键监听
Android Back Home键监听 Back键的监听 对于Back键的监听比较容易,可以在多个系统回调处拦截,比如在activity的下列方法中都可以收到Back键按下的事件: @Overrid ...
- Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听
原文:Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听 简单记录下android 盒子开发遥控器的监听 ,希望能帮到新入门的朋友们 不多说,直接贴代码 public cla ...
- Android之Home键监听封装
众所周知,我们监听返回键事件,无法是下面两个方法: @Override public void onBackPressed() { //do something //super.onBackPress ...
- flutter 返回键监听
本篇为继上片监听返回键基础下优化: 以下做返回键监听两种情况: import 'package:fluttertoast/fluttertoast.dart'; //提示第三方插件 1. 单击提示双击 ...
- Android 手势水平监听判断
package com.zihao.ui; import com.zihao.R; import android.os.Bundle; import android.app.Activity; imp ...
- Android中如何监听GPS开启和关闭
转自 chenming 原文 Android中如何监听GPS开启和关闭 摘要: 本文简单总结了如何监听GPS开关的小技巧 有时需要监听GPS的开关(这种需求并不多见).实现的思路是监听代表 GPS ...
随机推荐
- 带Cookie的 WebClient
/// <summary> /// WebClient的扩展 /// </summary> public class webClient : WebClient { /// & ...
- nova分析(10)—— nova-rootwrap
一.nova-rootwrap的作用 部署玩过openstack的都应该知道,它会生成一个nova用户来管理所有服务.nova身份在linux中属于普通用户级别,避免了一些需要root身份运行的操作, ...
- WPF性能提高--MSDN学习摘要
关于性能 一. 关于硬件加速 1.对于大多数图形硬件而言,大型图面是指达到 2048x2048 或 4096x4096 像素大小的图面. 二. 合理的布局 1.简单地说,布局是一个递归系统 ...
- js之引用类型
一.摘要: <javascript高级程序设计第三版>一书中单独有一章对js的引用类型(Object.Array.RegExp.Function:基本包装类型:Boolean.Number ...
- webstorm配置nodejs,bower,git,github
一,配置nodejs第一大步,首先安装nodejs,安装nodejs的时候,我们需要把所有的组建勾选上,然后选择add to path,这一步会自动帮我们配置环境变量,安装完成后,打开cmd,输入no ...
- CentOS 7.0体验与之前版本的不同
RHEL7和CentOS7出来有一段时间了,拿出点时间研究下,有几个地方跟6和5系列相比改变比较大,估计不少童鞋有点不太习惯.下面简要举例说明改变比较大的要点: 一.CentOS的Services使用 ...
- Puppet's Commands 3.7
Puppet's Commands Puppet’s command line interface consists of a single puppet command with many subc ...
- Tomcat的startup.bat一闪而过问题的解决
问题描述:点击Tomcat的startup.bat,一闪而过. 问题分析: 1.Tomcat的startup.bat--->catalina.bat--->setclasspath.bat ...
- activiti自定义流程之Spring整合activiti-modeler5.16实例(六):启动流程
注:(1)环境搭建:activiti自定义流程之Spring整合activiti-modeler5.16实例(一):环境搭建 (2)创建流程模型:activiti自定义流程之Spring ...
- SQL Server 中LEN函数的问题
LEN('T ') =1 LEN(' T') =2 在数据库中分解字符串时要注意,例如以'^'分隔'X ^ T ',分解时要注意最后的'T '被分解成'T' 可用如下的代码来进行完整的分解 SET A ...