当前位置: 首页 > news >正文

AR 眼镜之-普通电话-实现方案

目录

📂 前言

AR 眼镜之-蓝牙电话-实现方案

AR 眼镜系统版本

Android 手机系统版本

1. 🔱 技术方案

1.1 结构框图

1.2 方案介绍

1.3 实现方案

步骤一:手机 App 申请权限

步骤二:手机来电状态监听并推送给 AR 眼镜

步骤三:AR 眼镜显示来电信息并操作挂断/接听

步骤四:手机 App 执行挂断/接听操作

2. ⚛️ 自定义电话实现

2.1 自定义电话时序图

2.2 实现细节

1、手机 App 申请权限

2、手机来电状态监听并推送给 AR 眼镜

3、查询最近来电信息

4、手机 App 执行挂断/接听操作

5、API 监听电话状态调用

3. 💠 来电实现帮助类 TelephonyManagerHelper

4. ✅ 小结


📂 前言

AR 眼镜之-蓝牙电话-实现方案

AR 眼镜之-蓝牙电话-实现方案

AR 眼镜系统版本

        FreeRTOS。

Android 手机系统版本

        Android 15。

1. 🔱 技术方案

1.1 结构框图

1.2 方案介绍

  • 主要通过 BLE 定义私有协议,实现手机来电状态监听并推送给 AR 眼镜显示、以及在眼镜上实现接听和挂断电话的功能;

  • 与 BT 蓝牙电话不同的是,BLE 只能实现电话显示和控制功能,不能将通话音频传给 AR 眼镜,所以用户如果要接听电话,则需要通过手机扬声器或其他蓝牙耳机进行音频输出。

1.3 实现方案

步骤一:手机 App 申请权限

        申请手机来电状态权限 READ_PHONE_STATE、获取手机来电号码权限 READ_CALL_LOG、查询联系人名字权限 READ_CONTACTS、以及接/挂电话权限 ANSWER_PHONE_CALLS;

步骤二:手机来电状态监听并推送给 AR 眼镜
  1. 手机 App 监听到来电后,查询来电信息,包括:来电的电话号码以及联系人名字;

  2. 通过 BLE 定义的私有协议,将来电的电话号码以及联系人名字,推送给 AR 眼镜。

步骤三:AR 眼镜显示来电信息并操作挂断/接听
  1. RTOS AR 眼镜收到 BLE 私有协议命令后,调起来电 UI 界面,显示来电电话号码和名字;

  2. AR 眼镜将用户挂断或接听电话的操作,通过 BLE 命令发给手机 App。

步骤四:手机 App 执行挂断/接听操作

        手机 App 收到挂断/接听命令后,调用系统电话的挂断/接听接口。

2. ⚛️ 自定义电话实现

2.1 自定义电话时序图

2.2 实现细节

1、手机 App 申请权限

        1)在 Manifest 中申明权限

        申请手机来电状态权限 READ_PHONE_STATE、获取手机来电号码权限 READ_CALL_LOG、查询联系人名字权限 READ_CONTACTS、以及接/挂电话权限 ANSWER_PHONE_CALLS。

    <!--Phone Call Start--><!--手机来电状态--><uses-permission android:name="android.permission.READ_PHONE_STATE" /><!--获取手机来电号码--><uses-permission android:name="android.permission.READ_CALL_LOG" /><!--查询联系人名字--><uses-permission android:name="android.permission.READ_CONTACTS" /><!--接/挂电话--><uses-permission android:name="android.permission.ANSWER_PHONE_CALLS" /><!--Phone Call End-->

        2)检查权限

        如果权限缺失会发起请求,并在 Activity 的 onRequestPermissionsResult 中调用权限授予结果;否则,直接启动手机来电状态监听。

    private const val PERMISSIONS_REQUEST_CODE = 1000private val permissions = arrayOf(Manifest.permission.READ_PHONE_STATE,Manifest.permission.READ_CALL_LOG,Manifest.permission.READ_CONTACTS,Manifest.permission.ANSWER_PHONE_CALLS)private var applicationContext: Context? = null/** 检查权限,如果缺失会发起请求,否则直接启动监听 */fun checkPermissionsAndStart(context: Activity) {applicationContext = context.applicationContextval missingPermissions = permissions.filter {ContextCompat.checkSelfPermission(context, it) != PackageManager.PERMISSION_GRANTED}if (missingPermissions.isNotEmpty()) {ActivityCompat.requestPermissions(context, missingPermissions.toTypedArray(), PERMISSIONS_REQUEST_CODE)} else {startListener(context)}}/** 在 Activity 的 onRequestPermissionsResult 中调用 */fun onRequestPermissionsResult(requestCode: Int, grantResults: IntArray, context: Activity) {if (requestCode == PERMISSIONS_REQUEST_CODE) {if (grantResults.all { it == PackageManager.PERMISSION_GRANTED }) {startListener(context)} else {Toast.makeText(context,"Phone, call log, and call answer permissions are needed.",Toast.LENGTH_LONG).show()}}}
2、手机来电状态监听并推送给 AR 眼镜
  1. 手机 App 监听到来电后,查询来电信息,包括:来电的电话号码以及联系人名字;

  2. 通过 BLE 定义的私有协议,将来电的电话号码以及联系人名字,推送给 AR 眼镜。

    private var telephonyManager: TelephonyManager? = nullprivate val mPhoneListener = object : PhoneStateListener() {override fun onCallStateChanged(state: Int, phoneNumber: String?) {super.onCallStateChanged(state, phoneNumber)when (state) {TelephonyManager.CALL_STATE_IDLE -> {// 在注册监听的时候就会走一次回调,后面通话状态改变时也会走,如:在启动服务时如果手机没有通话相关动作,就会直接走一次TelephonyManager.CALL_STATE_IDLE。Log.i(TAG, "onCallStateChanged: 挂断 $phoneNumber")}TelephonyManager.CALL_STATE_OFFHOOK -> {Log.i(TAG, "onCallStateChanged: 接听 $phoneNumber")}TelephonyManager.CALL_STATE_RINGING -> {Log.i(TAG, "onCallStateChanged: 响铃 $phoneNumber")Log.e(TAG, "${getIncomingCallInfo(applicationContext?.contentResolver)}")}}}}/** 启动电话状态监听 */private fun startListener(context: Context) {Log.i(TAG, "startListener: ")telephonyManager =context.applicationContext.getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManagertelephonyManager?.listen(mPhoneListener, PhoneStateListener.LISTEN_CALL_STATE)}/** 停止监听 */fun stopListener() {Log.i(TAG, "stopListener: ")telephonyManager?.listen(mPhoneListener, PhoneStateListener.LISTEN_NONE)}
3、查询最近来电信息
    data class CallInfo(val number: String, val name: String = "")private fun getIncomingCallInfo(contentResolver: ContentResolver?): CallInfo? {var phoneNumber: String? = nullvar contactName = ""var cursor: Cursor? = nullvar nameCursor: Cursor? = nulltry {if (contentResolver != null) {// 查询最近一次来电号码cursor = contentResolver.query(CallLog.Calls.CONTENT_URI,arrayOf(CallLog.Calls.NUMBER),"${CallLog.Calls.TYPE} = ${CallLog.Calls.INCOMING_TYPE}",null,"${CallLog.Calls.DATE} DESC")cursor?.use {if (it.moveToFirst()) {phoneNumber = it.getString(it.getColumnIndexOrThrow(CallLog.Calls.NUMBER))}}// 查询联系人名字if (!phoneNumber.isNullOrEmpty()) {val uri: Uri = Uri.withAppendedPath(ContactsContract.PhoneLookup.CONTENT_FILTER_URI, Uri.encode(phoneNumber))nameCursor = contentResolver.query(uri, arrayOf(ContactsContract.PhoneLookup.DISPLAY_NAME), null, null, null)nameCursor?.use {if (it.moveToFirst()) {contactName =it.getString(it.getColumnIndexOrThrow(ContactsContract.PhoneLookup.DISPLAY_NAME))}}}}} catch (e: Exception) {e.printStackTrace()} finally {cursor?.close()nameCursor?.close()}return phoneNumber?.let { CallInfo(it, contactName) }}
4、手机 App 执行挂断/接听操作

        手机 App 收到挂断/接听命令后,调用系统电话的挂断/接听接口。

        1)接听电话

    /** 接听电话,内部自动检查权限 */fun answerCall(context: Context) {Log.i(TAG, "answerCall: ")if (ContextCompat.checkSelfPermission(context, Manifest.permission.ANSWER_PHONE_CALLS) == PackageManager.PERMISSION_GRANTED) {try {(context.getSystemService(Context.TELECOM_SERVICE) as TelecomManager).acceptRingingCall()} catch (e: Exception) {Log.e(TAG, "answerCall: 接听电话失败: ${e.message}")}} else {Log.e(TAG, "answerCall: 缺少 ANSWER_PHONE_CALLS 权限,无法接听!")}}

        2)挂断电话

    /** 挂断电话,内部自动检查权限 */fun endCall(context: Context): Boolean {Log.i(TAG, "endCall: ")var callSuccess = falsetry {if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {if (ActivityCompat.checkSelfPermission(context, Manifest.permission.ANSWER_PHONE_CALLS) == PackageManager.PERMISSION_GRANTED) {(context.getSystemService(Context.TELECOM_SERVICE) as TelecomManager).endCall()callSuccess = true} else {Log.e(TAG, "endCall: 缺少 ANSWER_PHONE_CALLS 权限,无法挂断!")}} else {val tm = context.getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManagerval m: Method = Class.forName(tm.javaClass.name).getDeclaredMethod("getITelephony")m.isAccessible = trueval telephonyService: ITelephony = m.invoke(tm) as ITelephonycallSuccess = telephonyService.endCall()Log.i(TAG, "endCall: 挂断电话成功 (低版本)!")}} catch (e: Exception) {Log.e(TAG, "endCall: ${e.printStackTrace()}")callSuccess = disconnectCall()e.printStackTrace()}return callSuccess}/** 挂断兜底方法,通过输入 keyevent */private fun disconnectCall(): Boolean {return try {Log.i(TAG, "disconnectCall: input keyevent " + KeyEvent.KEYCODE_ENDCALL)Runtime.getRuntime().exec("input keyevent " + KeyEvent.KEYCODE_ENDCALL.toString())true} catch (e: Exception) {Log.e(TAG, "disconnectCall: ${e.printStackTrace()}")false}}
5、API 监听电话状态调用
override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)// 启动电话监听(会自动检查权限)TelephonyManagerHelper.checkPermissionsAndStart(this)
}override fun onDestroy() {super.onDestroy()// 停止电话监听,释放资源TelephonyManagerHelper.stopListener()
}// 示例:来电时接听
fun answerIncomingCall() {TelephonyManagerHelper.answerCall(this)
}// 示例:挂断电话
fun hangupCall() {TelephonyManagerHelper.endCall(this)
}

3. 💠 来电实现帮助类 TelephonyManagerHelper

object TelephonyManagerHelper {data class CallInfo(val number: String, val name: String = "")private val TAG = TelephonyManagerHelper::class.java.simpleNameprivate const val PERMISSIONS_REQUEST_CODE = 1000private val permissions = arrayOf(Manifest.permission.READ_PHONE_STATE,Manifest.permission.READ_CALL_LOG,Manifest.permission.READ_CONTACTS,Manifest.permission.ANSWER_PHONE_CALLS)private var applicationContext: Context? = nullprivate var telephonyManager: TelephonyManager? = nullprivate val mPhoneListener = object : PhoneStateListener() {override fun onCallStateChanged(state: Int, phoneNumber: String?) {super.onCallStateChanged(state, phoneNumber)when (state) {TelephonyManager.CALL_STATE_IDLE -> {// 在注册监听的时候就会走一次回调,后面通话状态改变时也会走,如:在启动服务时如果手机没有通话相关动作,就会直接走一次TelephonyManager.CALL_STATE_IDLE。Log.i(TAG, "onCallStateChanged: 挂断 $phoneNumber")}TelephonyManager.CALL_STATE_OFFHOOK -> {Log.i(TAG, "onCallStateChanged: 接听 $phoneNumber")}TelephonyManager.CALL_STATE_RINGING -> {Log.i(TAG, "onCallStateChanged: 响铃 $phoneNumber")Log.e(TAG, "${getIncomingCallInfo(applicationContext?.contentResolver)}")}}}}/** 检查权限,如果缺失会发起请求,否则直接启动监听 */fun checkPermissionsAndStart(context: Activity) {applicationContext = context.applicationContextval missingPermissions = permissions.filter {ContextCompat.checkSelfPermission(context, it) != PackageManager.PERMISSION_GRANTED}if (missingPermissions.isNotEmpty()) {ActivityCompat.requestPermissions(context, missingPermissions.toTypedArray(), PERMISSIONS_REQUEST_CODE)} else {startListener(context)}}/** 在 Activity 的 onRequestPermissionsResult 中调用 */fun onRequestPermissionsResult(requestCode: Int, grantResults: IntArray, context: Activity) {if (requestCode == PERMISSIONS_REQUEST_CODE) {if (grantResults.all { it == PackageManager.PERMISSION_GRANTED }) {startListener(context)} else {Toast.makeText(context,"Phone, call log, and call answer permissions are needed.",Toast.LENGTH_LONG).show()}}}/** 启动电话状态监听 */private fun startListener(context: Context) {Log.i(TAG, "startListener: ")telephonyManager =context.applicationContext.getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManagertelephonyManager?.listen(mPhoneListener, PhoneStateListener.LISTEN_CALL_STATE)}/** 停止监听 */fun stopListener() {Log.i(TAG, "stopListener: ")telephonyManager?.listen(mPhoneListener, PhoneStateListener.LISTEN_NONE)}/** 接听电话,内部自动检查权限 */fun answerCall(context: Context) {Log.i(TAG, "answerCall: ")if (ContextCompat.checkSelfPermission(context, Manifest.permission.ANSWER_PHONE_CALLS) == PackageManager.PERMISSION_GRANTED) {try {(context.getSystemService(Context.TELECOM_SERVICE) as TelecomManager).acceptRingingCall()} catch (e: Exception) {Log.e(TAG, "answerCall: 接听电话失败: ${e.message}")}} else {Log.e(TAG, "answerCall: 缺少 ANSWER_PHONE_CALLS 权限,无法接听!")}}/** 挂断电话,内部自动检查权限 */fun endCall(context: Context): Boolean {Log.i(TAG, "endCall: ")var callSuccess = falsetry {if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {if (ActivityCompat.checkSelfPermission(context, Manifest.permission.ANSWER_PHONE_CALLS) == PackageManager.PERMISSION_GRANTED) {(context.getSystemService(Context.TELECOM_SERVICE) as TelecomManager).endCall()callSuccess = true} else {Log.e(TAG, "endCall: 缺少 ANSWER_PHONE_CALLS 权限,无法挂断!")}} else {val tm = context.getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManagerval m: Method = Class.forName(tm.javaClass.name).getDeclaredMethod("getITelephony")m.isAccessible = trueval telephonyService: ITelephony = m.invoke(tm) as ITelephonycallSuccess = telephonyService.endCall()Log.i(TAG, "endCall: 挂断电话成功 (低版本)!")}} catch (e: Exception) {Log.e(TAG, "endCall: ${e.printStackTrace()}")callSuccess = disconnectCall()e.printStackTrace()}return callSuccess}/** 挂断兜底方法,通过输入 keyevent */private fun disconnectCall(): Boolean {return try {Log.i(TAG, "disconnectCall: input keyevent " + KeyEvent.KEYCODE_ENDCALL)Runtime.getRuntime().exec("input keyevent " + KeyEvent.KEYCODE_ENDCALL.toString())true} catch (e: Exception) {Log.e(TAG, "disconnectCall: ${e.printStackTrace()}")false}}/** 查询最近来电信息 */private fun getIncomingCallInfo(contentResolver: ContentResolver?): CallInfo? {var phoneNumber: String? = nullvar contactName = ""var cursor: Cursor? = nullvar nameCursor: Cursor? = nulltry {if (contentResolver != null) {// 查询最近一次来电号码cursor = contentResolver.query(CallLog.Calls.CONTENT_URI,arrayOf(CallLog.Calls.NUMBER),"${CallLog.Calls.TYPE} = ${CallLog.Calls.INCOMING_TYPE}",null,"${CallLog.Calls.DATE} DESC")cursor?.use {if (it.moveToFirst()) {phoneNumber = it.getString(it.getColumnIndexOrThrow(CallLog.Calls.NUMBER))}}// 查询联系人名字if (!phoneNumber.isNullOrEmpty()) {val uri: Uri = Uri.withAppendedPath(ContactsContract.PhoneLookup.CONTENT_FILTER_URI, Uri.encode(phoneNumber))nameCursor = contentResolver.query(uri, arrayOf(ContactsContract.PhoneLookup.DISPLAY_NAME), null, null, null)nameCursor?.use {if (it.moveToFirst()) {contactName =it.getString(it.getColumnIndexOrThrow(ContactsContract.PhoneLookup.DISPLAY_NAME))}}}}} catch (e: Exception) {e.printStackTrace()} finally {cursor?.close()nameCursor?.close()}return phoneNumber?.let { CallInfo(it, contactName) }}}

4. ✅ 小结

        对于手机来电显示以及接听/挂断这块,本文只是一个基础实现方案,更多业务细节请参考产品逻辑去实现。

        另外,由于本人能力有限,如有错误,敬请批评指正,谢谢。

http://www.dtcms.com/a/577473.html

相关文章:

  • 下厨房网站学做蒸包视频可以上传自己做的视频的网站吗
  • IO卡常见问题处理
  • 11_FastMCP 2.x 中文文档之FastMCP高级功能:用户引导详解
  • 门户网站 移动端黄石专业网站建设推广
  • 类的嵌套 、封装
  • (128页PPT)麦肯锡金字塔原理培训思考写作和解决问题的逻辑(附下载方式)
  • Python 类实战:从“函数堆函数”到“客户端对象”,看类如何让 API 请求代码脱胎换骨
  • springboot的单元测试功能有什么用
  • 5昌平区网站建设免费模板网站哪个好
  • 济南网站制作设计公司网站建设属于什么科目
  • 深入解析Kafka的消息模型:如何确保消息不丢失且高效传递
  • 微服务之Nacos(注册中心、配置中心)
  • 导致Resources文件夹的资源在Android打包后丢失的原因
  • Leetcode 46
  • Zabbix 7 概述与配置详解
  • 网站优化体验报告中国创业网
  • 用 FastAPI + Pydantic 打造“可验证、可热载、可覆盖”的配置中心
  • 2025教资面试真题电子版|科目试讲+结构化真题解析|完整PDF
  • 一文了解-大语言模型训练 vs 推理:硬件算力需求数据对比
  • 影刀RPA一键分析用户行为!AI智能画像,转化率提升300%[特殊字符]
  • Spring Cache快速入门
  • 网站底部横条导航代码做网站的怎么挣钱、
  • 【科研绘图系列】R语言绘制散点图(scatter plot)
  • Supabase 概述
  • 【微服务】(3) 服务注册与发现
  • 网站综合查询工具做推文的编辑网站
  • Prometheus实战教程 05 - 告警通知实现 - 邮件 + 钉钉 + 自定义告警模板
  • SELinux 故障排除完全指南:从拒绝访问到快速修复
  • 【Linux】Socket编程预备及UDP
  • 建站运营新闻网页设计需要学什么学历