Android蓝牙线控切歌、连接状态监听(无线耳机也适用)
1. 监听蓝牙设备(音频)连接状态
所有代码已测试在Android11也能正常使用 (Android SDK 30)
首先新建一个广播类 BluetoothStateReceiver
/**
* @author komine
* 监听蓝牙音频设备的连接状态
*/
class BluetoothStateReceiver : BroadcastReceiver(){
override fun onReceive(context: Context?, intent: Intent?) {
when(intent?.action){
//蓝牙已连接,如果不需要判断是否为音频设备,下面代码块的判断可以不要
BluetoothDevice.ACTION_ACL_CONNECTED -> {
val device =
intent.getParcelableExtra<BluetoothDevice>(BluetoothDevice.EXTRA_DEVICE)
?: return
//如果在Android32 SDK还需要声明 android.permission.BLUETOOTH_CONNECT权限
if (isHeadPhone(device.bluetoothClass)) {
Log.d("BluetoothStateReceiver", "蓝牙音频设备已连接")
}
}
//蓝牙状态发生改变,断开,正在扫描,正在连接
BluetoothAdapter.ACTION_STATE_CHANGED -> {
val blueState = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, 0)
if (blueState == BluetoothAdapter.STATE_OFF) {
Log.d("BluetoothStateReceiver", "蓝牙设备已断开")
}
}
}
}
companion object{
private const val PROFILE_HEADSET = 0
private const val PROFILE_A2DP = 1
private const val PROFILE_OPP = 2
private const val PROFILE_HID = 3
private const val PROFILE_PANU = 4
private const val PROFILE_NAP = 5
private const val PROFILE_A2DP_SINK = 6
}
private fun isHeadPhone(bluetoothClass: BluetoothClass?): Boolean {
if (bluetoothClass == null) {
return false
}
return if (doesClassMatch(bluetoothClass, PROFILE_HEADSET)) true else doesClassMatch(bluetoothClass, PROFILE_A2DP)
}
//系统源码拷贝
//可以参考 http://androidxref.com/9.0.0_r3/xref/frameworks/base/core/java/android/bluetooth/BluetoothClass.java
private fun doesClassMatch(bluetoothClass: BluetoothClass, profile: Int): Boolean {
return if (profile == PROFILE_A2DP) {
if (bluetoothClass.hasService(BluetoothClass.Service.RENDER)) {
return true
}
when (bluetoothClass.deviceClass) {
BluetoothClass.Device.AUDIO_VIDEO_HIFI_AUDIO, BluetoothClass.Device.AUDIO_VIDEO_HEADPHONES, BluetoothClass.Device.AUDIO_VIDEO_LOUDSPEAKER, BluetoothClass.Device.AUDIO_VIDEO_CAR_AUDIO -> true
else -> false
}
} else if (profile == PROFILE_A2DP_SINK) {
if (bluetoothClass.hasService(BluetoothClass.Service.CAPTURE)) {
return true
}
when (bluetoothClass.deviceClass) {
BluetoothClass.Device.AUDIO_VIDEO_HIFI_AUDIO, BluetoothClass.Device.AUDIO_VIDEO_SET_TOP_BOX, BluetoothClass.Device.AUDIO_VIDEO_VCR -> true
else -> false
}
} else if (profile == PROFILE_HEADSET) {
// The render service class is required by the spec for HFP, so is a
// pretty good signal
if (bluetoothClass.hasService(BluetoothClass.Service.RENDER)) {
return true
}
when (bluetoothClass.deviceClass) {
BluetoothClass.Device.AUDIO_VIDEO_HANDSFREE, BluetoothClass.Device.AUDIO_VIDEO_WEARABLE_HEADSET, BluetoothClass.Device.AUDIO_VIDEO_CAR_AUDIO -> true
else -> false
}
} else if (profile == PROFILE_OPP) {
if (bluetoothClass.hasService(BluetoothClass.Service.OBJECT_TRANSFER)) {
return true
}
when (bluetoothClass.deviceClass) {
BluetoothClass.Device.COMPUTER_UNCATEGORIZED, BluetoothClass.Device.COMPUTER_DESKTOP, BluetoothClass.Device.COMPUTER_SERVER, BluetoothClass.Device.COMPUTER_LAPTOP, BluetoothClass.Device.COMPUTER_HANDHELD_PC_PDA, BluetoothClass.Device.COMPUTER_PALM_SIZE_PC_PDA, BluetoothClass.Device.COMPUTER_WEARABLE, BluetoothClass.Device.PHONE_UNCATEGORIZED, BluetoothClass.Device.PHONE_CELLULAR, BluetoothClass.Device.PHONE_CORDLESS, BluetoothClass.Device.PHONE_SMART, BluetoothClass.Device.PHONE_MODEM_OR_GATEWAY, BluetoothClass.Device.PHONE_ISDN -> true
else -> false
}
} else if (profile == PROFILE_HID) {
bluetoothClass.deviceClass and BluetoothClass.Device.Major.PERIPHERAL == BluetoothClass.Device.Major.PERIPHERAL
} else if (profile == PROFILE_PANU || profile == PROFILE_NAP) {
// No good way to distinguish between the two, based on class bits.
if (bluetoothClass.hasService(BluetoothClass.Service.NETWORKING)) {
true
} else bluetoothClass.deviceClass and BluetoothClass.Device.Major.NETWORKING == BluetoothClass.Device.Major.NETWORKING
} else {
false
}
}
}
然后在 MainActivity
中动态注册广播
class MainActivity : AppCompatActivity() {
private val mBluetoothStateReceiver:BluetoothStateReceiver = BluetoothStateReceiver()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
registerBluetoothReceiver()
}
private fun registerBluetoothReceiver(){
val intentFilter = IntentFilter()
intentFilter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED)
intentFilter.addAction(BluetoothDevice.ACTION_ACL_CONNECTED)
intentFilter.addAction(BluetoothDevice.ACTION_ACL_DISCONNECTED)
intentFilter.addAction("android.bluetooth.BluetoothAdapter.STATE_OFF")
intentFilter.addAction("android.bluetooth.BluetoothAdapter.STATE_ON")
registerReceiver(mBluetoothStateReceiver,intentFilter)
}
}
最后在 AndroidManifest.xml
声明权限
<uses-permission android:name="android.permission.BLUETOOTH"/>
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>
记得取消注册广播,不然打开其他音乐应用的时候,程序可能会crash
2. 监听蓝牙设备键按下
####################################2022-04-28更新############################
1.疑似Bug,需要在初始化MediaPlayer之后再注册才会生效,MediaButtonReceiver(applicationContext,this)放在MeidePlayer初始化之后代码
2.MediaSession只能有一个实例,如果在程序中创建了多个实例也会导致接收不到,其实看它命名也应该知道了...
####################################2022-04-28更新############################
要监听蓝牙设备的按键按下,你的应用需要处于音乐播放的状态,比如应用正在使用 MediaPlayer
播放音频,
如果未处于音频播放状态是不会触发回调方法.所以在演示代码中添加了MediaPlayer
来播放音频.
首先新建一个类MediaButtonReceiver类
class MediaButtonReceiver(mContent: Context, mKeyDownListener: IKeyDownListener) {
//创建一个MediaSession的实例,参数2是一个字符串,可以随便填
private val mMediaSession = MediaSession(mContent, javaClass.name)
internal annotation class KeyActions {
companion object {
//好像还支持手柄按键...
//所有keyCode参考:https://www.apiref.com/android-zh/android/view/KeyEvent.html
var PLAY_ACTION: Int = 126
var PAUSE_ACTION: Int = 127
var PREV_ACTION: Int = 88
var NEXT_ACTION: Int = 87
}
}
init {
mMediaSession.setCallback(object : MediaSession.Callback() {
override fun onMediaButtonEvent(mediaButtonIntent: Intent): Boolean {
val keyEvent: KeyEvent =
mediaButtonIntent.getParcelableExtra(Intent.EXTRA_KEY_EVENT)
?: return false
mKeyDownListener.onKeyDown(keyEvent.keyCode)
//返回值的作用跟事件分发的原理是一样的,返回true代表事件被消费,其他应用也就收不到了
return true
}
})
mMediaSession.setFlags(MediaSession.FLAG_HANDLES_MEDIA_BUTTONS or MediaSession.FLAG_HANDLES_TRANSPORT_CONTROLS)
mMediaSession.isActive = true
}
interface IKeyDownListener {
fun onKeyDown(@KeyActions keyAction: Int)
}
}
然后在 MainActivity
添加播放的代码
class MainActivity : AppCompatActivity(),MediaButtonReceiver.IKeyDownListener {
private val tag = javaClass.simpleName
private val mMediaPlay:MediaPlayer = MediaPlayer()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
mMediaPlay.reset()
val audioAttributesBuilder = AudioAttributes.Builder().setContentType(AudioAttributes.CONTENT_TYPE_MUSIC)
mMediaPlay.setAudioAttributes(audioAttributesBuilder.build())
//注意文件读写权限
mMediaPlay.setDataSource(Environment.getExternalStorageDirectory().toString() + File.separator + "1.ape")
mMediaPlay.setOnPreparedListener{
mMediaPlay.start()
MediaButtonReceiver(applicationContext,this)
}
mMediaPlay.prepareAsync()
}
override fun onKeyDown(keyAction: Int) {
when(keyAction){
MediaButtonReceiver.KeyActions.PLAY_ACTION -> Log.d(tag,"播放...")
MediaButtonReceiver.KeyActions.PAUSE_ACTION -> Log.d(tag,"暂停...")
MediaButtonReceiver.KeyActions.PREV_ACTION -> Log.d(tag,"上一首...")
MediaButtonReceiver.KeyActions.NEXT_ACTION -> Log.d(tag,"下一首...")
}
}
}
打印结果
参考:
Android 扫描蓝牙设备并获取设备类型
ボタンイベントの受信と処理
Android蓝牙线控切歌、连接状态监听(无线耳机也适用)的更多相关文章
- Android耳机线控具体解释,蓝牙耳机button监听(仿酷狗线控效果)
转载请注明出处:http://blog.csdn.net/fengyuzhengfan/article/details/46461253 当耳机的媒体按键被单击后.Android系统会发出一个广播.该 ...
- MTK Android 耳机线控的实现方法
android 耳机线控的实现方法 keycodeonkeydownkeyevent 耳机线控的功能 耳机线控是一种很好用,并且能提升用户体验的功能.可以用来实现一些常用和基本的功能.比如:实现音乐播 ...
- ZT android -- 蓝牙 bluetooth (五)接电话与听音乐
android -- 蓝牙 bluetooth (五)接电话与听音乐 分类: Android的原生应用分析 2013-07-13 20:53 2165人阅读 评论(9) 收藏 举报 蓝牙android ...
- Android USB大容量存储时SD卡状态监听(转)
对SD卡状态监听,到现在为止我知道的有两种方式: 1.注册StorageEventListener来监听sd卡状态 StorageEventListener中有onStorageStateChange ...
- 背水一战 Windows 10 (66) - 控件(WebView): 监听和处理 WebView 的事件
[源码下载] 背水一战 Windows 10 (66) - 控件(WebView): 监听和处理 WebView 的事件 作者:webabcd 介绍背水一战 Windows 10 之 控件(WebVi ...
- Android开发之手势滑动(滑动手势监听)详解
Android开发之手势滑动(滑动手势监听)详解 在Android应用中,经常需要手势滑动操作,比如上下滑动,或左右方向滑动,处理手势滑动通常有两种方法:一种是单独实现setOnTouchListen ...
- Android软键盘的隐藏显示、事件监听的代码
把开发过程中重要的一些内容片段做个珍藏,如下资料是关于Android软键盘的隐藏显示.事件监听的内容,应该是对小伙伴们有所用途. public class ResizeLayout extends L ...
- 截屏状态监听 - iOS
既接到电话状态监听的需求之后再次添加了截屏状态的监听,当使用 App 时若用户执行截屏操作需要对当前状态进行监听操作,下面有两种方法,其中可以替换截屏的图片内容(Plan A),也可以弹出提示框(Pl ...
- Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听
原文:Android TV开发中所有的遥控器按键监听及注意事项,新增home键监听 简单记录下android 盒子开发遥控器的监听 ,希望能帮到新入门的朋友们 不多说,直接贴代码 public cla ...
随机推荐
- Centos使用crontab自动定时备份mysql的脚本
在我们网站上线之后免不了需要备份数据库,为什么要备份呢?我给大家列出了3个理由. 1.防止数据丢失 2.防止数据改错了,可以用来恢复 3.方便给客户数据 以 上几点告诉我们要经常备份,当然我今天给大家 ...
- 使用cmd命令行执行MySQL数据库
说明 用命令提示符来操作一些简单的数据库,便捷又快速,随便记录一下,以后没事就自己来看看! 哈哈哈! 打开/关闭mysql服务 net start mysql net stop mysql 连接mys ...
- 老子云AMRT全新三维格式正式上线,其性能全面超越现有的三维数据格式
9月16日,老子云AMRT全新三维格式正式上线,其性能远超现有的三维数据格式.目前已有含国家超算长沙中心.中科院空间所.中车集团等上百家政企事业单位的项目中使用了AMRT格式,大大提升了可视化项目的开 ...
- 不存在的!python说不给数据的浏览器是不存在的!
有时候我们些代码是总发此疑惑? 为什么别人采集 xx 网站的时候能成功,而我却总是不返回给数据出现这种原因时往往是我们没有给够伪装, 被识别了出来~ 就像人,你出门肯定是要穿衣服的对不,如果你不穿! ...
- docker安装node
#1.拉取镜像 docker pull node:latest #2.运行 docker run -itd --name node-test --restart=always node #--rest ...
- window下Redis快速启动,以及闪退问题解决
我下载的是免安装版的window版redis,解压后如下: 第一步:在解压的redis文件夹下新建一个redis-start.bat(window启动一般都是xx.bat) 第二步:打开redis.w ...
- 选择结构-穿透的switch语句和循环结构-循环概述
case的穿透性 在switch语句中,如果case的后面不写break,将出现穿透现象,也就是不会在判断下一个case的值,直接向后运 行,直到遇到break,或者整体switch结束 publi ...
- Windows 远程连接后,自动断开,所有程序都自动关闭(待验证,待更新)
win+r输入regedit打开注册表编辑SecurityLayer,将值改为2 计算机\HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Ter ...
- Java开发学习(十四)----Spring整合Mybatis及Junit
一.Spring整合Mybatis思路分析 1.1 环境准备 步骤1:准备数据库表 Mybatis是来操作数据库表,所以先创建一个数据库及表 create database spring_db cha ...
- 了解有哪几个C标准&了解C编译管道
下列哪个不是C标准.参考:C语言标准 小知识:C语言标准的发展 K&R C: 1978年,丹尼斯·里奇(Dennis Ritchie)和布莱恩·科尔尼干(Brian Kernighan)出版了 ...