转载请注明出处:http://blog.csdn.net/fengyuzhengfan/article/details/46461253

当耳机的媒体按键被单击后。Android系统会发出一个广播。该广播的携带者一个Action名为MEDIA_BUTTON的Intent。监听该广播便能够获取手机的耳机媒体按键的单击事件。

在Android中有个AudioManager类,该类会维护MEDIA_BUTTON广播的分发。所以要实现耳机按键监听须要向AudioManager注冊一个用于接收耳机按键单击事件的接收器:

AudioManager audioManager = (AudioManager)context
.getSystemService(Context.AUDIO_SERVICE);
ComponentName name = newComponentName(context.getPackageName(),
MediaButtonReceiver.class.getName());
audioManager.registerMediaButtonEventReceiver(name);

该方法的原型为:

publicvoid registerMediaButtonEventReceiver (PendingIntent eventReceiver)

Added in API level 18

Registera component to be the sole receiver of MEDIA_BUTTON intents. This is like registerMediaButtonEventReceiver(android.content.ComponentName),
but allows the buttons to go to any PendingIntent. Note that you shouldonly use this form if you know you will continue running for the full timeuntil unregistering the PendingIntent.

Parameters

eventReceiver

target that will receive media button intents. The PendingIntent will be sent an ACTION_MEDIA_BUTTON event
when a media button action occurs, with EXTRA_KEY_EVENT added
and holding the key code of the media button that was pressed.

从API凝视中可知:

1、 在AudioManager对象注冊一个MediaoButtonRecevie,使它成为MEDIA_BUTTON的唯一接收器,也就是说仅仅有我能收到,其它的都收不到这个广播了,否则的话大家都收到会照成一定的混乱;

2、该广播必须在AndroidManifest.xml文件里进行声明。否则就监听不到该MEDIA_BUTTON广播了。

注,由于当我们注冊了AudioManager媒体按键的接收器。而且该接收器是媒体按键的唯一接收器。所以要在不使用按键监听的时候取消该注冊:

AudioManager audioManager = (AudioManager)context    .getSystemService(Context.AUDIO_SERVICE);
ComponentName name = newComponentName(context.getPackageName(), MediaButtonReceiver.class.getName());
audioManager.unregisterMediaButtonEventReceiver(name);

当耳机媒体键发生单击事件的时候Android系统会发出两次广播,第一次是按键按下去的时候,第二次是松开按键的时候,为了可以准确的获知用户单击了几次媒体键,我们仅仅须要在按键松开的时候处理单击事件就可以:

KeyEvent keyEvent = (KeyEvent)intent.getParcelableExtra(Intent.EXTRA_KEY_EVENT); //获得KeyEvent对象
if(keyEvent.getAction()== KeyEvent.ACTION_UP){
//在这里处理单击事件
}

以下就分别解说一下为了实现线控效果所用到的几个类:

1.   耳机线控管理工具类HeadSetUtil:

package com.jph.lc;

import android.content.ComponentName;
import android.content.Context;
import android.media.AudioManager;
import android.util.Log;
/**
* 耳机线控管理工具类 单例
* @author JPH
* @date 2015-6-9 下午4:03:45
*/
public class HeadSetUtil { private static HeadSetUtil headSetUtil;
private OnHeadSetListener headSetListener = null; public static HeadSetUtil getInstance() {
if (headSetUtil == null) {
headSetUtil = new HeadSetUtil();
}
return headSetUtil;
} /**
* 设置耳机单击双击监听接口 必须在open前设置此接口,否则设置无效
* @param headSetListener
*/
public void setOnHeadSetListener(OnHeadSetListener headSetListener) {
this.headSetListener = headSetListener;
} /**
* 为MEDIA_BUTTON 意图注冊接收器(注冊开启耳机线控监听, 请务必在设置接口监听之后再调用此方法,否则接口无效)
* @param context
*/
public void open(Context context) {
if(headSetListener==null){
throw new IllegalStateException("please set headSetListener");
}
AudioManager audioManager = (AudioManager) context
.getSystemService(Context.AUDIO_SERVICE);
ComponentName name = new ComponentName(context.getPackageName(),
MediaButtonReceiver.class.getName());
audioManager.registerMediaButtonEventReceiver(name);
Log.i("ksdinf", "open");
}
/**
* 关闭耳机线控监听
* @param context
*/
public void close(Context context) {
AudioManager audioManager = (AudioManager) context
.getSystemService(Context.AUDIO_SERVICE);
ComponentName name = new ComponentName(context.getPackageName(),
MediaButtonReceiver.class.getName());
audioManager.unregisterMediaButtonEventReceiver(name);
}
/**
* 删除耳机单机双击监听接口
*/
public void delHeadSetListener() {
this.headSetListener = null;
} /**
* 获取耳机单击双击接口
*
* @return
*/
protected OnHeadSetListener getOnHeadSetListener() {
return headSetListener;
} /**
* 耳机button单双击监听
*/
public interface OnHeadSetListener {
/**
* 单击触发,主线程。 此接口真正触发是在单击操作1秒后 由于须要推断1秒内是否仍监听到点击,有的话那就是双击了
*/
public void onClick();
/**
* 双击触发,此接口在主线程。能够放心使用
*/
public void onDoubleClick();
/**
* 三连击
*/
public void onThreeClick();
}
}

该类主要负责媒体按键接收器的注冊和自己定义媒体按键回调监听器的设置。

该类中包括一个OnHeadSetListener接口,该接口中的onClick(),onDoubleClick()。onThreeClick()三个方法分别会在单击事件。双击事件,以及三连击事件发生时被回调。

须要指出的是。单击和双击事件会有1秒的延迟,这是由于在这1秒内须要监听是否还有单击发生的原因,当然这1s也不是绝对的,你也能够依据实际的业务须要自己定义事件。

在以下解说的这个类中将会解开酷狗线控的原理。

2.耳机媒体按键广播接收器MediaButtonReceiver:

package com.jph.lc;

import java.util.Timer;
import java.util.TimerTask; import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.KeyEvent; import com.jph.lc.HeadSetUtil.OnHeadSetListener; /**
* MEDIA_BUTTON耳机媒体按键广播接收器
* @author JPH
* @Date2015-6-9 下午8:35:40
*/
public class MediaButtonReceiver extends BroadcastReceiver{ private Timer timer = null;
private OnHeadSetListener headSetListener = null;
private static MTask myTimer = null;
/**单击次数**/
private static int clickCount;
public MediaButtonReceiver(){
timer = new Timer(true);
this.headSetListener = HeadSetUtil.getInstance().getOnHeadSetListener();
}
@Override
public void onReceive(Context context, Intent intent) {
Log.i("ksdinf", "onReceive");
String intentAction = intent.getAction() ;
if(Intent.ACTION_MEDIA_BUTTON.equals(intentAction)){
KeyEvent keyEvent = (KeyEvent)intent.getParcelableExtra(Intent.EXTRA_KEY_EVENT); //获得KeyEvent对象
if(headSetListener != null){
try {
if(keyEvent.getAction() == KeyEvent.ACTION_UP){
if (clickCount==0) {//单击
clickCount++;
myTimer = new MTask();
timer.schedule(myTimer,1000);
}else if (clickCount==1) {//双击
clickCount++;
}else if (clickCount==2) {//三连击
clickCount=0;
myTimer.cancel();
headSetListener.onThreeClick();
}
}
} catch (Exception e) {
}
}
}
abortBroadcast();//终止广播(不让别的程序收到此广播,免受干扰)
}
/**
* 定时器,用于延迟1秒,推断是否会发生双击和三连击
*/
class MTask extends TimerTask{
@Override
public void run() {
try {
if (clickCount==1) {
mhHandler.sendEmptyMessage(1);
}else if (clickCount==2) {
mhHandler.sendEmptyMessage(2);
}
clickCount=0;
} catch (Exception e) {
// TODO: handle exception
}
}
};
/**
* 此handle的目的主要是为了将接口在主线程中触发
* ,为了安全起见把接口放到主线程触发
*/
Handler mhHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
if(msg.what==1){//单击
headSetListener.onClick();
}else if (msg.what==2) {//双击
headSetListener.onDoubleClick();
}else if (msg.what==3) {//三连击
headSetListener.onThreeClick();
}
}
}; }

该类主要负责接收系统发出的媒体案件的单击事件。并对单击事件做对应的处理以达到单击。双击。三连击的效果。须要指出的是该类在实例化的时候会获取OnHeadSetListener监听器,所以要在调用HeadSetUtil类的open方法用之前设置OnHeadSetListener。否则将不会对媒体按键事件做处理。

该类中有个名为Mtask的内部类,该内部类是一个定时任务,该任务会在指定的时间里分析是否会发生双击和三连击。

另外,该类中另一个myHandler对象,该对象是为了将回调监听发生在UI线程中,以方便UI的更新。

3.监听器的使用类MainActivity:

package com.jph.lc;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.widget.TextView; import com.jph.lc.HeadSetUtil.OnHeadSetListener;
/**
* 耳机线控实例。蓝牙耳机button监听(仿酷狗线控效果)
* @author JPH
* @Date2015-6-10 上午9:49:02
*/
public class MainActivity extends Activity { TextView txt = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
txt = (TextView) findViewById(R.id.text);
HeadSetUtil.getInstance().setOnHeadSetListener(headSetListener);
HeadSetUtil.getInstance().open(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
HeadSetUtil.getInstance().close(this);
}
OnHeadSetListener headSetListener = new OnHeadSetListener() {
@Override
public void onDoubleClick() {
txt.setText("双击");
Log.i("ksdinf", "双击");
}
@Override
public void onClick() {
txt.setText("单击");
Log.i("ksdinf", "单击");
}
@Override
public void onThreeClick() {
txt.setText("三连击");
Log.i("ksdinf", "三连击");
}
};
}

该类中举要介绍了媒体按键监听的使用。

源代码下载:http://download.csdn.net/detail/fengyuzhengfan/8797357

GitHub:https://github.com/crazycodeboy/HeadSetControl

Android耳机线控具体解释,蓝牙耳机button监听(仿酷狗线控效果)的更多相关文章

  1. Second Day: 关于Button监听事件的三种方法(匿名类、外部类、继承接口)

    第一种:通过匿名类实现对Button事件的监听 首先在XML文件中拖入一个Button按钮,并设好ID,其次在主文件.java中进行控件初始化(Private声明),随后通过SetOnClickLis ...

  2. 仿酷狗音乐播放器开发日志二十三 修复Option控件显示状态不全的bug(附源码)

    转载请说明原出处,谢谢~~ 整个仿酷狗工程的开发将近尾声,现在还差选项设置窗体的部分,显然在设置窗体里用的最多的就是OptionUI控件,我在写好大致的布局后去测试效果,发现Option控件的显示效果 ...

  3. Android开发之手势滑动(滑动手势监听)详解

    Android开发之手势滑动(滑动手势监听)详解 在Android应用中,经常需要手势滑动操作,比如上下滑动,或左右方向滑动,处理手势滑动通常有两种方法:一种是单独实现setOnTouchListen ...

  4. Android软键盘的隐藏显示、事件监听的代码

    把开发过程中重要的一些内容片段做个珍藏,如下资料是关于Android软键盘的隐藏显示.事件监听的内容,应该是对小伙伴们有所用途. public class ResizeLayout extends L ...

  5. 背水一战 Windows 10 (66) - 控件(WebView): 监听和处理 WebView 的事件

    [源码下载] 背水一战 Windows 10 (66) - 控件(WebView): 监听和处理 WebView 的事件 作者:webabcd 介绍背水一战 Windows 10 之 控件(WebVi ...

  6. Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听

    原文:Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听 简单记录下android 盒子开发遥控器的监听 ,希望能帮到新入门的朋友们 不多说,直接贴代码 public cla ...

  7. Android修行之路------List view无法获取监听方法

    注意: 1.在list view自定义布局中如果添加滚动布局,会导致自定义布局无法获取监听. 2.如果ListView的每项布局里有像Button,ImageButton之类View的控键时,这些Vi ...

  8. Android的事件处理机制详解(二)-----基于监听的事件处理机制

    基于监听的事件处理机制 前言: 我们开发的app更多的时候是需要与用户的交互----即对用户的操作进行响应 这就涉及到了android的事件处理机制; android给我们提供了两套功能强大的处理机制 ...

  9. UGUI研究院之控件以及按钮的监听事件系统

    继续学习,我相信大家在做NGUI开发的时候处理事件都会用到UIEventListener,那么UGUI中怎么办呢?先看UGUI的事件有那些吧 Supported Events The Eventsys ...

随机推荐

  1. sqlserver还原3101

    1.出现错误"3101" 2.解决办法:删除数据库之后还原(有风险)或者获得数据库的独占访问权(用sql语句) 参考:https://www.2cto.com/database/2 ...

  2. C指针类型转换问题

    先看下面的代码: #include<stdio.h> int main () { int a; char *x; x = (char *) &a; a = 512; x[0] = ...

  3. Openjudge-4132-四则运算表达式求值

    这一题我们可以通过递归求解,首先我们可以把一个表达式分为三部分,分别是: (1)表达式 :项.加减 (2)项:因子.乘除 (3)因子:数.()表达式 这三项构成了递归的关系,我们可以看到,要求一个表达 ...

  4. 关于 vertical-align

    默认情况下(行内基线位置 = 行内元素最大高度): 如果对这个正方形使用 vertival-align:middle.在最大高度的元素上使用负值(middle = - 50% * 元素高度),可以提升 ...

  5. Django之web本质

    Django之web本质 Web的本质,是基于socket玩的. 在我们上网的过程中,一个访问请求是如何工作的. Web的框架: 网络的连接都是基于Socket 在连接中有TCP/UDP 和HTTP协 ...

  6. 《算法导论》 — Chapter 7 快速排序

    序 快速排序(QuickSort)也是一种排序算法,对包含n个数组的输入数组,最坏情况运行时间为O(n^2).虽然这个最坏情况运行时间比较差,但是快速排序通常是用于排序的最佳实用选择,这是因为其平均性 ...

  7. Couchbase IV(管理与维护)

    Couchbase IV(管理与维护) 管理 常用命令 Command Description server-list List all servers in a cluster server-inf ...

  8. div添加滚动条

  9. XV6上下文切换

    上下文切换分为两种情况 用户程序陷入到内核,再从内核返回 两个应用程序之间的上下文切换 用户程序陷入到内核 用户程序陷入到内核通过中断INT指令,在xv6中系统调用的号为64 操作系统在初始化的时候会 ...

  10. NYOJ-481平衡字符串

    平衡字符串 时间限制:1000 ms  |  内存限制:65535 KB 难度:2 描述 给你一定长度的字符串.字符串中只包含26个小写字母,首先我们把字母a-z分为2堆(a--m)和(n--z),判 ...