Android 系统提供了支持 SIP(Session Initiation Protocol)的 API,允许开发者添加基于 SIP 的因特网电话特性到自己的应用程序中。
Android 包含一个完整的 SIP 协议栈,整合了允许轻松创建来电和去电的电话管理服务,而不必开发者直接参与管理会话、传输层通信、音频录制等工作。
目前 SIP 已经被成功应用于视频会议和即时消息中,其应用程序开发需要基于 Android 2.3(API Level 9)以上的系统。
SIP 运行于无线数据连接,通过 AVD 无法调试。在 SIP 应用程序通信会话中,每一个参与者都必须拥有一个 SIP 账号。
Android SDK 中与 SIP 开发相关的类和接口被放置在 android.net.sip 包中,相关类和接口介绍如下。
| 类/接口 | 作用 | 
|---|---|
| SipAudioCall | 用于处理基于 SIP 的因特网音频呼叫。 | 
| SipAudioCall.Listener | 用于处理 SIP 呼叫事件,如接收到呼叫和对外呼叫事件。 | 
| SipErrorCode | 定义了 SIP 行为的错误代码。 | 
| SipManager | 提供了 SIP 任务的相关 API,例如初始化 SIP 连接,提供对相关 SIP 服务的访问等。 | 
| SipProfile | 定义了一个 SIP 配置文件,包括 SIP 账户、域和服务器信息等。 | 
| SipProfile.Builder | 创建 SIP 配置信息的帮助类。 | 
| SipSession | 代表一个与 SIP 对话框相关联的 SIP 会话或者一个单独的无对话框的事务。 | 
| SipSession.Listener | 针对 SIP 会话事件的监听器,例如会话被注册或者一个电话正在呼出事件。 | 
| SipSession.State | 定义了 SIP 会话的状态信息,例如注册、呼出、呼入等。 | 
| SipRegistrationListener | 一个用于监听 SIP 注册事件的接口。 | 
要开发基于 SIP 的应用程序,必须使用 Android 2.3 以上版本的设备,但是并不是所有 Android 2.3 以上版本的设备都支持 SIP 应用程序。
为应用程序添加 SIP 支持需要在应用程序的配置文件 AndroidManifest.xml 中添加如下内容。
1)添加使用 SIP 和因特网的权限:
	<uses-permission android:name="android.permission.USE_SIP" />
	<uses-permission android:name="android.permission.INTERNET" />
2)确保应用程序只可以被安装在支持 SIP 的设备上,在 Manifest 文件中添加以下代码:
	<uses-sdk android:minSdkVersion="9" />
	<uses-feature android:name="android.hardware.sip.voip" />
3)如果应用程序被设计为接收呼叫,则必须定义一个 receiver:
<receiver android:name=".IncomingCallReceiver" android:label="Call Receiver"/>
应用程序的 Manifest 文件示例代码如下:
<?xml version="l.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.android.sip">
    ...
    <receiver android:name=".IncomingCallReceiver" android:label="Call Receiver"/>
    ...
    <uses-sdk android:minSdkVersion="9" />
    <uses-permission android:name="android.permission.USE_SIP" />
    <uses-permission android:name="android.permission.INTERNET" />
    ...
    <uses-feature android:name="android.hardware.sip.voip" android:required="true" />
    <uses-feature android:name="android.hardware.wifi" android:required="true" />
    <uses-feature android:name="android.hardware.microphone" android:required="true" />
</manifest>
要使用 SIP API,必须创建 SipManager 示例。SipManager 用于处理:
创建 SipManager 对象的代码如下:
public SipManager mSipManager = null;
...
if(mSipManager == null){
    mSipManager = SipManager.newInstance(this);
}
在典型的 Android SIP 应用程序中包含一个或多个用户,每个用户都必须有一个 SIP 账户。在 SIP 应用程序中,SIP 账户用 SipProfile 对象表示。
SipProfile 定义了 SIP 配置简表,包括 SIP 账户以及域和服务器信息。与运行应用程序的设备上的 SIP 账户相关联的配置简表叫作本地简表,会话连接到的简表叫作对等简表。
当 SIP 应用程序使用本地 SipProfile 登录到 SIP 服务器时,SipProfile 帮助 SIP 服务器高效地将当前设备注册为 SIP 呼叫的目的地。
创建 SipProfile 对象的代码如下:
public SipProfile mSipProfile = null; ... SipProfile.Bulider bulider = new SipProfile(username,domain); bulider.setPassword(password); mSipProfile = builder.bulid();
下面代码中的 SipManager 打开本地简表,用于拨打或者接收 SIP 呼叫:
Intent intent - new Intent();
intent.setAction("android.SipDemo.INCOMING_CALL");
PendingIntent pendingIntent = PendingIntent.getBroadcast(this,0,intent,Intent.FILL_IN_DATA);
mSipManager.open(mSipProfile,pendingIntent,null);
下面的代码为 SipManager 注册了 SipRegistrationListener 接口,该接口用于跟踪 SipProfile 是否在 SIP 服务提供者处成功注册。
SipManager.setRegistrationListener (mSipProfile.getUriString(), new SipRegistrationListener(){
    public void onRegistering (String localProfileUri) {
        updateStatus ("Registering with SIP Server...");
    }
    public void onRegistrationDone (String localProfileUri, long expiryTime){
        updateStatus ("Ready");
    }
   
    public void onRegistrationFailed (String localProfileUri, int errorCode, String errorMessage) {
        updateStatus ("Registration failed. Please check settings.");
    }
下面的代码演示了配置简表使用结束后,如何关闭简表,并从服务器注销设备信息。
public void closeLocalProfile(){
    if (mSipManager==null) {
        return;
    }
    try {
        if (mSipProfile !=null) {
            mSipManager.close (mSipProfile.getUriString());
        }
    } catch (Exception e) {
        Log.d ("WalkieTalkieActivity/onDestroy", "Failed to close local profile.", e);
    }
}
要使用 SIP 拨打语音电话,需要满足如下条件:
为了拨打音频电话,需要创建 SipAudioCall.Listener 对象。
大部分的客户端与 SIP 栈之间的交互都是通过接口进行的。下面的代码演示了呼叫建立后接口如何进行处理:
SipAudioCall.Listener listener=new SipAudioCall.Listener(){
  
    @Override
    public void onCallEstablished (SipAudioCall call) {
        call.startAudio();
        call.setSpeakerMode (true);
        call.toggleMute();
        ...
    }
    @Override
    public void onCallEnded (SipAudioCall call) {
        // Do something.
    }
};
SipAudioCall.Listener 接口创建后,通过 SipManager.makeAudioCall() 方法进行音频呼叫。该方法有 4 个参数,分别是:
进行音频呼叫的代码如下:
call=mSipManager.makeAudioCall(mSipProfile.getUriString(),sipAddress, listener, 30);
为了接收呼叫,SIP 应用程序必须包含一个 BroadcastReceiver 的子类,以便当有来电时用于对 Intent 对象进行处理。为此,需要在应用程序中完成以下几步:
1)在 AndroidManifest.xml 文件中声明 <receiver>,例如:
<receiver android:name=".IncomingCallReceiver" android:label="Call Receiver"/>2)
2)实现声明的 BroadcastReceiver 的子类,例如 IncomingCallReceiver。
3)使用 PendingIntent 对象初始化本地 SipProfile。当有来电时,该PendingIntent 会启动 BroadcastReceiver 的子类。
4)设置 Intent Filter,用于过滤来电时产生的 Intent。
下面的代码定义了一个用于处理来电的 BroadcastReceiver:
/*** Listens for incoming SIP calls, intercepts and hands them off to WalkieTalkieActivity. */
public class IncomingCallReceiver extends BroadcastReceiver {
    /**
     *Processes the incoming call, answers it, and hands it over to the
     *WalkieTalkieActivity.
     *@param context The context under which the receiver is running.
     *Sparam intent The intent being received.
     */
    @Override
    public void onReceive (Context context, Intent intent) {
        SipAudioCall incomingCall=null;
        try {
            SipAudioCall.Listener listener=new SipAudioCall.Listener(){
            @Override
            public void onRinging (SipAudioCall call, SipProfile caller) {
            try {
                call.answerCall (30) ;
            } catch (Exception e) {
                e.printStackTrace ();
            }
        }
    };
    WalkieTalkieActivity wtActivity= (WalkieTalkieActivity) context; 
    incomingCall=wtActivity.mSipManager.takeAudioCall (intent, listener);
    incomingCall.answerCall (30);
    incomingCall.startAudio();
    incomingCall.setSpeakerMode (true);
    if (incomingCall.isMuted()) {
        incomingCall.toggleMute();
    }
    wtActivity.call=incomingCall; wtActivity.updateStatus (incomingCall);
    } catch (Exception e) {
        if (incomingCall !=null) {
            incomingCall.close();
        }
    }
}
设置用于接收来电的 Intent Filter 的相关代码如下:
public SipManager mSipManager=null;
public SipProfile mSipProfile=null;
...
Intent intent=new Intent();
intent.setAction ("android.SipDemo.INCOMING_CALL"); PendingIntent pendingIntent=PendingIntent.getBroadcast (this, 0,intent,Intent.FILL_IN_DATA);
mSipManager.open (mSipProfile, pendingIntent, null);
Intent Filter 信息可以在应用程序的 Manifest 文件中被注册,也可以像下面的代码演示的那样在 Activity 的 onCreate() 方法中被注册。相关代码如下:
public class WalkieTalkieActivity extends Activity implements View.OnTouchListener {
    ...
    public IncomingCallReceiver callReceiver;
    ...
    @Override
    public void onCreate (Bundle savedInstanceState) {
        IntentFilter filter=new IntentFilter();
        filter.addAction ("android.SipDemo.INCOMING_CALL");
        callReceiver=new IncomingCallReceiver();
        this.registerReceiver (callReceiver, filter);
        ...
    }
    ...
}
					更多...
加载中...