转载请注明出处: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. Python自动化测试框架——断言

    在自动化测试执行的过程中,我们往往希望可以自定生成报告,那如何再测试中进行验证呢?我们使用断言 import unittest class TestCount(unittest.TestCase): ...

  2. (11) openssl req(生成请求证书、私钥和自建CA)

    伪命令req大致有3个功能:生成证书请求文件.验证证书请求文件和创建根CA. 由于openssl req命令选项较多,所以先各举几个例子,再集中给出openssl req的选项说明.若已熟悉opens ...

  3. Linux文件查找find和locate

    目 录 第1章 locate文件查找    1 1.1 概述    1 1.2 locate文件查找的特性    1 第2章 文件查找概述    1 第3章    1 3.1 文件名查找    1 3 ...

  4. Python运维工程师

    1.单引号,双引号,三引号的区别. 2.Python里面如何实现tuple和List的转换. 3.Python的参数传递是值传递还是引用传递,举例说明Python函数参数传递的几种形式,并说明函数传参 ...

  5. 主题:学习Spring必学的Java基础知识(9)----HTTP报文

    转: 引述要学习Spring框架的技术内幕,必须事先掌握一些基本的Java知识,正所谓“登高必自卑,涉远必自迩”.以下几项Java知识和Spring框架息息相关,不可不学(我将通过一个系列分别介绍这些 ...

  6. Area(poj 1654)

    题目大意:一个坐标系,从原点开始走,然后1-4分别代表,向右下走,向右走,向右上走,向下走,5代表回到原点,6-9代表,向上走,向左下走,向左走,向左上走.给出一串包含1-9的字符串,问你这些点所围成 ...

  7. hdu 2736 Average distance

    传送门 Average distance Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Oth ...

  8. Git学习之常见错误 git push 失败

    Git学习之常见错误 git push 失败 问题描述: git push Counting objects: , done. Delta compression using up to thread ...

  9. 匿名函数--lambda函数

    匿名函数 匿名函数:为了解决一些功能很简单的需求而设计的一句话函数 (python对匿名函数支持有限,只有一些简单的条件下可以用匿名函数) 匿名函数固定格式: func = lambda *args: ...

  10. 【Perl】perl正则表达式中的元字符、转义字符、量词及匹配方式

    Linux平台上被广泛使用的正则表达式库PCRE - Perl-compatible regular expressions,从其名字即可知道,PCRE提供的是一套与Perl中相兼容的正则表达式. 元 ...