转载请注明出处:http://blog.csdn.net/vnanyesheshou/article/details/71429860
继续研究hfp相关功能。蓝牙耳机可以控制手机接听、拒接、挂断电话,拨打电话等功能。本文主要分析下起这些操作的大致流程。
在系统应用Bluetooth中com_android_bluetooth.cpp提供了多个回调方法,由hardware、协议栈回调过来。蓝牙耳机的一些控制命令都会发到这里。
本文基于Android4.3源码。
1 接通电话##
蓝牙耳机控制手机接通电话,回掉com_android_bluetooth.cpp中的answer_call_callback()函数,该函数主要操作是调用HeadsetStateMachine的onAnswerCall()函数,代码如下:
在onAnswerCall()中发送消息(消息类型STACK_EVENT,StackEvent事件类型EVENT_TYPE_ANSWER_CALL)向状体机,此时通话尚未接通,audio没有连接,所以此时处于Connected状态。状态机收到该消息后调用processAnswerCall()函数。processAnswerCall()代码如下:
private void processAnswerCall() {
if (mPhoneProxy != null) {
try {
//mPhoneProxy是通过bindservice 获取的。
mPhoneProxy.answerCall();
} catch (RemoteException e) {
}
} else {
}
}
初始化的时候会bind service,绑定的该service为系统应用Phone下的BluetoothPhoneService(AndroidManifest中该service的action为android.bluetooth.IBluetoothHeadsetPhone),代码如下:
//参数为android.bluetooth.IBluetoothHeadsetPhone
Intent intent = new Intent(IBluetoothHeadsetPhone.class.getName());
//resolveSystemService该方法是hide的,由系统使用的特殊功能来解决系统应用程序的服务意图。
intent.setComponent(intent.resolveSystemService(context.getPackageManager(), 0));
if (intent.getComponent() == null || !context.bindService(intent, mConnection, 0)) {
Log.e(TAG, "Could not bind to Bluetooth Headset Phone Service");
}
绑定service成功回调mConnection,在其成功回调中设置的mPhoneProxy。通过mPhoneProxy来调用service中提供的接口。mPhoneProxy.answerCall()跳到BluetoothPhoneService中answerCall。
public boolean answerCall() {
//申请权限,修改电话状态
enforceCallingOrSelfPermission(MODIFY_PHONE_STATE, null);
return PhoneUtils.answerCall(mCM.getFirstActiveRingingCall());
}
PhoneUtils调用answerCall,在这里面去接通电话。answerCall()就不具体分析了。
2 拒接、挂断电话##
蓝牙耳机控制手机拒接、挂断电话,回掉com_android_bluetooth.cpp中的hangup_call_callback()函数,该函数主要操作是调用HeadsetStateMachine的onHangupCall()函数,代码如下:
private void onHangupCall() {
StackEvent event = new StackEvent(EVENT_TYPE_HANGUP_CALL);
sendMessage(STACK_EVENT, event);
}
此时HeadsetStateMachine可能处于Conneted或AudioOn状态,这两种状态收到该消息的处理一样,都是调用processHangupCall(),代码如下:
private void processHangupCall() {
if (isVirtualCallInProgress()) {
//对于虚拟电话,结束。
terminateScoUsingVirtualVoiceCall();
} else {
if (mPhoneProxy != null) {
try { //挂断电话
mPhoneProxy.hangupCall();
} catch (RemoteException e) {
}
} else {
}
}
}
对于虚拟电话则直接将其结束。真实的通话跳到BluetoothPhoneService的hangupCall。
public boolean hangupCall() {
enforceCallingOrSelfPermission(MODIFY_PHONE_STATE, null);
if (mCM.hasActiveFgCall()) { //挂断正在进行的通话
return PhoneUtils.hangupActiveCall(mCM.getActiveFgCall());
} else if (mCM.hasActiveRingingCall()) { //停止正在响铃的电话
return PhoneUtils.hangupRingingCall(mCM.getFirstActiveRingingCall());
} else if (mCM.hasActiveBgCall()) { //挂断保持的电话
return PhoneUtils.hangupHoldingCall(mCM.getFirstActiveBgCall());
}
return false;
}
hangupCall中会根据状态处理通话,优先处理正在进行的通话、其次是尚未接通的电话、最后是保持的电话。
3 更改通话音量##
蓝牙耳机更改通话的音量,回掉com_android_bluetooth.cpp中的volume_control_callback()函数,该函数主要操作是调用HeadsetStateMachine的onVolumeChnaged()函数,代码如下:
private void onVolumeChanged(int type, int volume) {
StackEvent event = new StackEvent(EVENT_TYPE_VOLUME_CHANGED);
event.valueInt = type;
event.valueInt2 = volume;
sendMessage(STACK_EVENT, event);
}
此时HeadsetStateMachine可能处于Conneted或AudioOn状态,这两种状态收到该消息的处理一样,都是调用processVolumeEvent,代码如下:
private void processVolumeEvent(int volumeType, int volume) {
if (volumeType == HeadsetHalConstants.VOLUME_TYPE_SPK) {
mPhoneState.setSpeakerVolume(volume);
//是否在ui上显示
int flag = (getCurrentState() == mAudioOn) ? AudioManager.FLAG_SHOW_UI : 0;
//设置SCO通道声音大小。
mAudioManager.setStreamVolume(AudioManager.STREAM_BLUETOOTH_SCO, volume, flag);
} else if (volumeType == HeadsetHalConstants.VOLUME_TYPE_MIC) {
// 只是存了下该volume值,并没有设置mic。
mPhoneState.setMicVolume(volume);
} else {
}
}
更改音量两种类型,VOLUME_TYPE_MIC类型,保存了下该值,并没有看到具体用该值的地方。对于VOLUME_TYPE_SPK类型的,会设置SCO声音大小。如果此时处于AudioOn状态,则会在UI上显示。
4 拨打电话##
蓝牙耳机进行拨打电话,回掉com_android_bluetooth.cpp中的dial_call_callback函数,该函数主要操作是调用HeadsetStateMachine的onDialCall()函数,代码如下:
private void onDialCall(String number) {
StackEvent event = new StackEvent(EVENT_TYPE_DIAL_CALL);
event.valueString = number;
sendMessage(STACK_EVENT, event);
}
此时HeadsetStateMachine可能处于Conneted或AudioOn状态,这两种状态收到该消息的处理一样,都是调用processDialCall,代码如下:
private void processDialCall(String number) {
String dialNumber;
if ((number == null) || (number.length() == 0)) {
//获取最近向外打的电话号码
dialNumber = mPhonebook.getLastDialledNumber();
if (dialNumber == null) { //没有最近拨打的电话,回应error
atResponseCodeNative(HeadsetHalConstants.AT_RESPONSE_ERROR, 0);
return;
}
} else if (number.charAt(0) == '>') {
//测试
} else {
// Remove trailing ';'
if (number.charAt(number.length() - 1) == ';') {
number = number.substring(0, number.length() - 1);
}
dialNumber = PhoneNumberUtils.convertPreDial(number);
}
terminateScoUsingVirtualVoiceCall(); // 终止虚拟呼叫
Intent intent = new Intent(Intent.ACTION_CALL_PRIVILEGED,
Uri.fromParts(SCHEME_TEL, dialNumber, null));
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
mService.startActivity(intent); //开启拨打电话的界面
mDialingOut = true;
sendMessageDelayed(DIALING_OUT_TIMEOUT, DIALING_OUT_TIMEOUT_VALUE);
}
蓝牙耳机发过来的命令可能携带电话号码,也可能不带,对于没有电话号码则查询最近的拨打电话记录,拨打最近拨打的电话。对于有号码,则拨打该号码。
Intent.ACTION_CALL_PRIVILEGED(该变量是hide的,执行任何号码的呼叫,紧急或不紧急):"android.intent.action.CALL_PRIVILEGED"
通过该action打开系统应用Phone中的OutgoingCallBroadcaster界面,向外进行拨打电话。
欢迎大家关注、评论、点赞。
你们的支持是我坚持的动力。