Android 蓝牙开发(八)hfp接听、挂断电话

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

本文已授权微信公众号 fanfan程序媛 独家发布 扫一扫文章底部的二维码或在微信搜索 fanfan程序媛 即可关注

继续研究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界面,向外进行拨打电话。

欢迎扫一扫关注我的微信公众号,定期推送优质技术文章:

这里写图片描述

发布了129 篇原创文章 · 获赞 270 · 访问量 97万+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章