电话应用程序和打电话流程

     android电话应用程序概述

      一.总览

        1.phone应用主界面

        2.拨出电话流程

        3.接入电话流程

     二.phone应用主界面

        phone应用程序的主界面包括”phone”,”Call log“,“Contacts”,”Favorites”四个部分组成。

        Phone:由12键的拨号盘组成。输入号码后单击电话图标按钮进行拨号

        Call log:存放呼入和呼出的历史记录。

        Contacts:存放电话联系人。

        Favorites:存放常用的电话联系人。

      

       (1)以上4个部分由“DialtactsActivity.java”作为主要入口。在onCreate()方法中通hasPhoneProviderExtras过“setupDalerTab()”,“setupCallLogTab()”,”setupContactsTab”,”setupFavoritestTab()”以上4个方法构建以“tabHost”形式显示4个部分的主界面入口。而以上4个功能界面的实现分别由“TwelveKeyDialer.java”,”RecentListActity.java”,”ContactsListActivity.java”,3个部分实现。在这里,我们重点描述“Phone”部分的实现。

     三.拨出电话流程  

    进入由”TwelveKeyDIaler”构建的电话拨号程序,输入呼出的电话号码,单击拨号按钮。拨号的实现由placeCall()中实现。

Intent intent = new Intent(Intent.ACTION_CALL_PRIVILEGED,

                Uri.fromParts("tel", number, null));

        if (number == null || !TextUtils.isGraphic(number)) {

            // There is no number entered.

            if (phoneIsCdma() && phoneIsOffhook()) {

                // We only want to send this empty flash extra if we're CDMA and the

                // phone is offhook (don't want to send if ringing or dialing)

                intent.putExtra(EXTRA_SEND_EMPTY_FLASH, true);

                sendEmptyFlash = true;

            } else {

                playTone(ToneGenerator.TONE_PROP_NACK);

                return;

            }

        }

        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

        startActivity(intent);

        通过查找”Intent.ACTION_CALL_PRIVILEGED”的实际字符窜,得知intent跳转的类为OutgoingCallBroadCaster.java.在此类的onCreate()方法中,对呼出的电话号码进行判断,并把号码定义分类为“紧急号码”和“非紧急号码”  :如果是紧急号码,则通过intent.setClass(this,InCallScreen.class);startActivity(intent)执行;如果是非紧急号码,则通过new Intent(Intent.ACTION_NEW_OUTGOING_CALL);sendOrderedBroadcast();跳转到内部类OutgoingCallReceiver.java中执行拨号处理。

      在“OutgoingCallReceiver”中“doReceiver()”对电话网络进行判断是否为cdma网络,如果是进一步判断是否基于OTA呼叫(这点不是很明白?)。此后跳转至IncallScreen类,进行呼叫处理。

      “IncallScreen”类构建了电话呼出界面以及对呼出的处理和响应。如果是第一次呼叫则执行onCreate(),非第一次则执行onNewIntent().此后这两个方法中都拥有internalResolveIntent(intent),此方法获得电话呼叫状态,其中包括了呼出状态和呼入状态。其中的placecall()方法返回得到呼出状态,返回结果由IncallScreen中的“InCallInitStatus”枚举变量定义。注:通过在placecall()中的phone.dial()建立与framework层的电话拨号连接。

       四.电话呼入流程

        首先,呼叫通知由phoneApp来实现。

        PhoneApp.java->onCreate()->CallNotifier.java->handleMessage()->onNewRingingConnection()接受来自ril层的电话响应,并发出电话铃音或者震动。

        当电话发出铃音或震动的同时,继续执行PhoneUtils.java->showIncomingCallUi()->PhoneApp.java->void displayCallScreen()中的startActivity()跳转到InCallScreen.java中执行初始化呼入界面。

       继续执行internalResolveIntent(intent)->internalAnswerCall()->PhoneUtils.answerCall() ->Phone.accept()为电话开始接听

 

2. 

Android打电话流程

1 OutgoingCallBroadcaster.java
点击拨号盘拨打按钮后,进入通话的Phone包。此时首先进入的函数是OutgoingCallBroadcaster.java,该类是一个Activity。
由activity的生命周期可知,第一次进入时应调用onCreate()函数。(在该类中也只实现了这个函数。)解析一下这个函数:
1.1) 首先获取Intent对象,获取拨出的号码。
1.2) 接着判断号码是否为紧急号码,如果是紧急号码,将callNow变量赋值为true,启动InCallScreen,并发送广播。而在receiver里面判断callNow为ture就直接finish,而不再重复启动InCallScreen;如果不是紧急号码,将callNow变量赋值为false,发送广播“Intent.ACTION_NEW_OUTGOING_CALL”。

2 OutgoingCallReceiver.java
广播发送后OutgoingCallReceiver将会收到该息。该类是一个内部类,在类OutgoingCallBroadcaster里面,作用是接收OutgoingCallBroadcaster发送的广播,判断是否已经启动InCallScreen。没有启动的话就进行一些初始化,如:对OTA进行初始化。接收到广播之后,从Intent里面取出电话号码及其URi。然后,设置Intent为ACTION_CALL,并带上号码和uri。启动InCallScreen。关闭该Activity。
OTA:Over-the-Air Technology空中下载技术,是通过移动通信(GSM或CDMA)的空中接口对SIM卡数据及应用进行远程管理的技术。空中接口可以采用WAP、 GPRS、CDMA1X及短消息技术。OTA技术的应用,使得移动通信不仅可以提供语音和数据服务,而且还能提供新业务下载。
GSM:Global System for Mobile Communications,中文为全球移动通讯系统,俗称"全球通"。
CDMA:Code Division Multiple Access,又称码分多址,是在无线通讯上使用的技术,CDMA允许所有的使用者同时使用全部频带(1.2288Mhz),并且把其他使用者发出的讯号视为杂讯,完全不必考虑到讯号碰撞 (collision) 的问题。CDMA的优点包括:CDMA中所提供的语音编码技术,其通话品质比目前的GSM好,而且可以把用户对话时周围环境的噪音降低,使通话更为清晰。

3 InCallScreen.java
该类extends了Acitivity,并且implements了 OnClickListener,OnTouchListener和OnQueryCompleteListener。该类主要是负责通话的那一个界面,并且还负责菜单项各种按键事件和触摸时间的处理。同时本类还复写的finish()方法,所以一般不会被finish掉,调用这个方法时它又把自己放回栈中。InCallScreen可以接收这个Intent并启动。
3.1) onCreate(第一次)
3.1.1) callScreenOnCreate获得通话界面被创建的时间。
3.1.2) PhoneApp唤醒后台的服务程序。
3.1.3) 判断当前的通话状态(IDLE =没有通话行为,RINGING =正在通话或呼叫等待,OFFHOOK = The phone is off hook. At least one call exists that is dialing, active or holding and no calls are ringing or waiting.),如果正在通话,不会出现键盘锁。接着设置mPhone and mForegroundCall/mBackgroundCall/mRingingCall。
3.1.4) getBluetoothHandsfree设置蓝牙耳机,如果存在蓝牙耳机,则安装该设备。
3.1.5) initInCallScreen加载各种view组建。
3.1.6) 对通话的各种状态进行广播。(registerForPreciseCallStateChanged,registerForDisconnect,registerForMmiInitiateregisterForMmiComplete,registerForCallWaiting,registerForSuppServiceFailed,registerForCdmaOtaStatusChange)
3.1.7) internalResolveIntent判断是否使用了OTA技术,通过该判断设置通话界面的样式。
3.1.8) callScreenCreated记录通话界面创建完成后的时间
3.2) onNewIntent(非第一次)。
我们重新启动一个Intent时调用该函数。由于我们围绕唯一的一个InCallScreen实例来完成通过的这个过程,那么除了第一次被创建的InCallScreen,只要有来电或者去电,该程序就会发生。如果InCallScreen已经在前台,该程序也会发生。
3.2.1) setIntent保存该Intent,以至于将来我们可以获得该intent。
3.2.2) internalResolveIntent
3.3) onResume
进行一些初始化操作,如:获取一个PhoneApp对象,解开Keyguard Notification的statusBar给Disable。还内置了一个Handler可以回调处理一些事件,比如:PHONE_STATE_CHANGED,PHONE_DISCONNECT,EVENT_HEADSET_PLUG_STATE_CHANGED。同时有一个独立的BroadcastReceiver处理ACTION_HEADSET_PLUG,比如插入耳机等。
3.3.1) 首先对锁屏情况下的来电除了处理。
3.3.2) disableStatusBar当正在通话界面时,使得状态栏可用。
3.3.3) setIgnoreTouchUserActivity忽略通话过程中无意的触碰事件,使得这些无意的触碰不会阻止设备进入休眠。
3.3.4) registerReceiver监听广播
3.3.5) startDialerSession当在前台是,保持一个dialer session。首先判断时候需要播放本地铃声,如果需要,则判断双音多频是否可用,如果可用,则创建一个声音播放器。
3.3.6) isBluetoothAudioConnected做一个是否蓝牙连接的判断。
3.3.7) 如果是cdma通话,则初始化OTA状态,进而如果是采用了OTA,则设置InCallScreenMode为OTA通话模型。
3.3.8) clearDisconnected在检查该通话状态之前,切断其他网络连接。
3.3.9) syncWithPhoneState同步通话界面与Phone的当前状态。如果没有同步成功,则dismissAllDialogs();结束当前的所以通话,endInCallScreenSession();关闭通话界面的显示。
3.3.10) updateWakeState设置基于当前Phone的唤醒状态和屏幕超时,以及通话界面的当前状态。
3.3.11) enableTouchLock当onresume时“触摸锁“叠加是不可见的,尤其是这个检查可确保用户通话按MUNU来唤醒屏幕后将不会被锁。但如果拨号盘是打开的,而又需要通话计时,则造就了“触摸锁“覆盖。

4 Profiler.java
该类对通话各个时间点进行记录。
static long sTimeCallScreenRequested;//通话界面被请求的时间
static long sTimeCallScreenOnCreate;//通话界面被创建的时间
static long sTimeCallScreenCreated;//通话界面创建完成后的时间
static long sTimeIncomingCallPanelRequested;//正在通话时,通话界面被请求的时间
static long sTimeIncomingCallPanelOnCreate;//正在通话时,通话界面被创建的时间
static long sTimeIncomingCallPanelCreated;//正在通话时,通话界面创建完成后的时间

5 PhoneUtils.java
负责Phone对象的生成,主要调用phone.getForegroundCall(),phone.getBackgroundCall(),phone.getRingingCall(),phone.dial()。
5.1) placeCall(Phone phone, String number, Uri contactRef)
拨打传入的电话号码,该函数被InCallScreen中的placeCall调用。
参数phone手机对象;参数number用户要拨打的号码;参数contactRef要么是“tel:”,要么是 “content://contacts”,取决于通话初始化,该参数引发呼叫;返回 CALL_STATUS_DIALED,CALL_STATUS_DIALED_MMI,或CALL_STATUS_FAILED。
5.2) placeCallVia(Context context,Phone phone,String number,Uri contactRef,Uri gatewayUri)
使用第三方提供的网关拨打号码,该函数被InCallScreen中的placeCall调用。如果电话号码是紧急号码,GSM MMI码或者CDMA码则不能被呼叫。如果连接成立,这个方法发出一个同步调用阻止查询来电信息,使本地采用异步查询。
参数phone手机对象;参数context执行CallerInfo查询;参数number用户要拨打的号码,如果号码不能建立连接,则仅被用于建立电话卡,并更新通话记录;参数contactRef要么是“tel:”,要么是“content://contacts”,该参数引发呼叫;参数 gatewayUri用于设置连接的地址;返回CALL_STATUS_DIALED或CALL_STATUS_FAILED。

6 PhoneApp.java
该类是一个普通的java类,主要负责Phone对象顶层应用的生成。这是一个虚拟的Phone对象,它从 framework层取得一个Phone对象。该类继承自Application,同时能常驻内存,他和PhoneUtils一起处理电话操作。在 oncreate方法里面进行各种全局的初始化:获取Phone对象 NotificationMgr对象,PowerManager对象,SimCard对象等。同时内置的Handler可以回调处理各种事件,如:EVENT_SIM_ABSENT,EVENT_SIM_NETWORK_LOCKED,EVENT_UPDATE_INCALL_NOTIFICATION 等。

7 CallNotifier.java
监听Phone状态的改变和来自telephony层各种事件,并触发任何有关的UI行为(如开始的铃声和来电的用户界面,打在通话音,更新通知,写呼叫记录条目等)。
7.1) 在构造函数中,实现
mPhone.registerForNewRingingConnection(this, PHONE_NEW_RINGING_CONNECTION, null);
mPhone.registerForPreciseCallStateChanged(this, PHONE_STATE_CHANGED, null);
mPhone.registerForDisconnect(this, PHONE_DISCONNECT, null);
mPhone.registerForUnknownConnection(this, PHONE_UNKNOWN_CONNECTION_APPEARED, null);
另外如果是CDMA通讯类型,还执行:
mPhone.registerForCdmaOtaStatusChange(this, EVENT_OTA_PROVISION_CHANGE, null);
mPhone.registerForCallWaiting(this, PHONE_CDMA_CALL_WAITING, null);
mPhone.registerForDisplayInfo(this, PHONE_STATE_DISPLAYINFO, null);
mPhone.registerForSignalInfo(this, PHONE_STATE_SIGNALINFO, null);
mPhone.registerForInCallVoicePrivacyOn(this, PHONE_ENHANCED_VP_ON, null);
mPhone.registerForInCallVoicePrivacyOff(this, PHONE_ENHANCED_VP_OFF, null);
mPhone.registerForCdmaFwdBurstDtmf(this, PHONE_CDMA_FWD_BURST_DTMF, null);
mPhone.registerForCdmaFwdContDtmfStart(this, PHONE_CDMA_FWD_CONT_DTMF_START, null);
mPhone.registerForCdmaFwdContDtmfStop(this, PHONE_CDMA_FWD_CONT_DTMF_STOP, null);
如果是GSM类型,则执行:
mPhone.registerForRingbackTone(this, PHONE_RINGBACK_TONE, null);
mPhone.registerForResendIncallMute(this, PHONE_RESEND_MUTE, null);
7.2) 整个类由函数handleMessage串起来。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值