android -- phone (二) 去电流程

        这篇是关于外拨电话的具体流程,也就是去电流程,虽然网上的资料很多(重复的也很多),但作为电话的主要操作之一,为了保证phone系列的完整性,还是要把它写一下的。开始看代码。

TwelveKeyDialer.java,既然要打电话,总要先输入号码才拨出,这个类就是拨号盘的界面,只是这个phone用到的类却是放在com.android.contacts包下,应该是出于代码结构的考虑吧。毕竟联系人、拨号盘、通话记录和收藏都是在一个Tab标签里的。这个界面没什么好说的,0-9数字键,P和W(也可能是*和#),P表示直接拨打带有分机号的号码(如2345-0000P1234)时会直接拨分机号,无需要再输入分机号码,而W拨号(如23450000W1234)则会有对话框提示你确认是否拨分机号,就这点区别。

       按完电话号码点拨号键,接下来就开始去电流程了,离开TwelveKeyDialer.java前的代码,

    void dialButtonPressed(String ipPrefix) {
        Log.d(TAG, "dialButtonPressed");   
        //注意这个action,只有从拨号盘拨出的才是Intent.ACTION_CALL_PRIVILEGED
        //外部调用的是Intent.ACTION_CALL
        Intent intent = new Intent(Intent.ACTION_CALL_PRIVILEGED);  
        //省略次要代码……
        intent.setData(Uri.fromParts("tel", number, null));//拼一个电话的Uri,关键代码
        StickyTabs.saveTab(this, getIntent());//与拨号无关
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        startActivity(intent);
        mDigits.getText().clear(); //清除输入
     }

         发出来的Intent谁来接收呢,是OutgoingCallBroadcaster.java类,这是一个中间类,实际我们是看不到它的,这个类先判断号码是否为紧急号码,如果是紧急号码,启动InCallScree.java,并发送广播;如果不是,发送广播“Intent.ACTION_NEW_OUTGOING_CALL”由它的内部类OutgoingCallReceiver接收,从onReceiver()再到doReceiver(),会把action里的字符串统一替换成Intent.ACTION_CALL,启动InCallScreen.java,在这个类也可以做一些其它的业务逻辑判断(比如固定拨号,视频电话等),IncallScreen.java是电话应用的主界面,这个界面负责的东西比较多,所需要做的判断也不少,值得一提的是它重写finish()方法,当调用这个方法时它又把自己放回栈中,这样可提高下次启动的响应速度。

    如果是第一次进入IncallScreen,会执行onCreate()
    protected void onCreate(Bundle icicle) {
        Profiler.callScreenOnCreate();//获得通话界面被创建的时间。
        …..省略代码...       
        setPhone(app.phone);  // Sets mPhone
        mCM =  PhoneApp.getInstance().mCM;
        mBluetoothHandsfree = app.getBluetoothHandsfree();//设置蓝牙
        if (mBluetoothHandsfree != null) {           
            mBluetoothHeadset = new BluetoothHeadset(this, null); 
        }
        initInCallScreen();   //加载界面元素
     …..省略代码...   
        registerForPhoneStates();//注册各种状态
        if (icicle == null) {            
    mInCallInitialStatus = internalResolveIntent(getIntent());  //这个是关键代码         
        } else {
            mInCallInitialStatus = InCallInitStatus.SUCCESS;
        }
        mUseTouchLockOverlay = !app.proximitySensorModeEnabled();
        Profiler.callScreenCreated();//记录通话界面创建完成后的时间
    }
    如果是第二次进入会执行onNewIntent()
    protected void onNewIntent(Intent intent) {      
        mInCallInitialStatus = internalResolveIntent(intent);//这个是关键代码
        ……去掉log代码
    }

        两个方法都会走到internalResolveIntent(intent),这里我们关心电话拨出的动作是怎么跑下去的,所以InCallScreen里的onResume()方法就不细看了,那里面有关于锁屏和蓝牙连接等逻辑判断。回来继续看internalResolveIntent()的代码。

    InCallInitStatus internalResolveIntent(Intent intent) {
        ……省略很多代码…   传过来的action是Intent.ACTION_CALL,直接看重点
        } else if (action.equals(Intent.ACTION_CALL)
                || action.equals(Intent.ACTION_CALL_EMERGENCY)) {
            app.setRestoreMuteOnInCallResume(false);
            if (PhoneUtils.hasPhoneProviderExtras(intent)) {               
            InCallInitStatus status = placeCall(intent);//这是我们要找的
            if (status == InCallInitStatus.SUCCESS) {        
                app.setBeginningCall(true);
            }
            return status;
        } }

        进入placeCall()方法后,会做一些关于是否紧急号码和OTA(Over-the-Air Technology空中下载技术,是通过移动通信(GSM或CDMA)的空中接口对SIM卡数据及应用进行远程管理的技术)的判断,不过我们更关心下面的代码。

    if (null != mProviderGatewayUri && !(isEmergencyNumber || isEmergencyIntent) &&
         PhoneUtils.isRoutableViaGateway(number)) {  
         callStatus = PhoneUtils.placeCallVia(this, phone, number,
           contactUri, mProviderGatewayUri);//多了个网关参数
    } else {
         callStatus = PhoneUtils.placeCall(phone, number, contactUri);//多数调用这里
    }

       后面switch分支会根据callStatus的值完成相应的功能或提示。到这里到代码就走到PhoneUtils.java里了,placeCall()是一个静态方法调用。里面最重要的代码是     

    Connection cn = PhoneApp.getInstance().mCM.dial(phone, number);
    //也有可能看到下面的代码
    Connection cn = phone.dial(number)//如果是这种,会先走到GsmCallTracker.java

        顺便提下在这个方法里还会有setAudioMode(),关于音频通道和模式的设置就在这里,这里面的故事也不少,不过要先放放了。下面的代码已经在Framework层了,不管之前的代码在哪里dial,最后都会来到RIL.java。        

     public void dial(String address, int clirMode, UUSInfo uusInfo, Message result) {
        RILRequest rr = RILRequest.obtain(RIL_REQUEST_DIAL, result);
        rr.mp.writeString(address);
        rr.mp.writeInt(clirMode);
        rr.mp.writeInt(0); // UUS information is absent

        if (uusInfo == null) {
            rr.mp.writeInt(0); // UUS information is absent
        } else {
            rr.mp.writeInt(1); // UUS information is present
            rr.mp.writeInt(uusInfo.getType());
            rr.mp.writeInt(uusInfo.getDcs());
            rr.mp.writeByteArray(uusInfo.getUserData());
        } 
        send(rr);
    }

       跟着RIL_REQUEST_DIAL这个TAG标志向下走来到Reference-ril.c找到相应的case分支,

   case RIL_REQUEST_DIAL:
            requestDial(data, datalen, t);

   static void requestDial(void *data, size_t datalen, RIL_Token t){
    p_dial = (RIL_Dial *)data;

    switch (p_dial->clir) {
        case 1: clir = "I"; break;  /*invocation*/
        case 2: clir = "i"; break;  /*suppression*/
        default:
        case 0: clir = ""; break;   /*subscription default*/
    }
    asprintf(&cmd, "ATD%s%s;", p_dial->address, clir); 
    ret = at_send_command(cmd, NULL);
    free(cmd);  
    RIL_onRequestComplete(t, RIL_E_SUCCESS, NULL, 0);
   }

      到这里我们的电话基本上算是拨出去了,至于是否成功还要看返回结果,通常形式上的流程是这样的(下面只是很多可能中的一种,也是比较常见的一种),

    ===>>[SendAT] ATD15812345678 //拨号,看到atd是不是会想到贝尔实验室和AT&T呢?
    <<====[RecvAT] OK
    <<====[RecvAT] +CLCC: 1, 0, 2, 0, 0, "15812345678", 129, //主动上报clcc
    ===>>[SendAT] AT+CMUT=0, time=   //设置话简静音关
    <<====[RecvAT] OK, time=
     ===>>[SendAT] AT+CLCC           //主动查询,
    <<====[RecvAT] +CLCC: 1, 0, 2, 0, 0, "15812345678", 129, //以这一次的clcc为准
        这段代码描述这样一个事实,不管模块报上来CLCC数据如何,我们都会重新查询后再上报给应用层,上层收到消息后更新界面和维护Connection里每一路电话的状态。到这一步,去电的流程也算是基本走完了,后面还有些界面状态刷新的代码省略了,对于AT命令那一部分,具体实现要看芯片厂商,表现上不完全一致。关于去电的流程就写这些,尽量写详细点,不过也有点长了,如仍有遗漏的地方,欢迎补充。


 


    









你可能感兴趣的:(android -- phone (二) 去电流程)