Core-Telecom 程式庫提供一套穩健且一致的 API,可簡化將通話應用程式與 Android 平台整合的程序
如要瞭解實際的實作方式,請前往 GitHub 查看範例應用程式:
- 輕量型範例應用程式:這個最簡單的範例會示範
Core-TelecomAPI 的用法。非常適合快速瞭解基本概念。 - 完整範例應用程式 (由 Core-Telecom 團隊開發):功能更豐富的應用程式,展示進階電信功能和最佳做法。這項資源非常適合用來瞭解複雜的整合情境。
設定 Core-Telecom
在應用程式的 build.gradle 檔案中新增 androidx.core:core-telecom 依附元件:
dependencies {
implementation ("androidx.core:core-telecom:1.0.0")
}
在 AndroidManifest.xml 中宣告 MANAGE_OWN_CALLS 權限:
<uses-permission android:name="android.permission.MANAGE_OWN_CALLS" />
註冊應用程式
使用 CallsManager 向 Android 註冊通話應用程式,開始將通話新增至系統。註冊時,請指定應用程式的功能 (例如音訊、影片支援):
val callsManager = CallsManager(context)
val capabilities: @CallsManager.Companion.Capability Int =
(CallsManager.CAPABILITY_BASELINE or
CallsManager.CAPABILITY_SUPPORTS_VIDEO_CALLING)
callsManager.registerAppWithTelecom(capabilities)
通話管理
使用 Core-Telecom API 建立及管理通話生命週期。
建立通話
CallAttributesCompat 物件會定義專屬通話的屬性,可具有下列特徵:
displayName:來電者姓名。address:通話地址 (例如電話號碼、Meet 會議連結)。direction:來電或外撥。callType:音訊或視訊。callCapabilities:支援轉接和保留。
以下範例說明如何建立來電:
fun createIncomingCallAttributes(
callerName: String,
callerNumber: String,
isVideoCall: Boolean): CallAttributesCompat {
val addressUri = Uri.parse("YourAppScheme:$callerNumber")
// Define capabilities supported by your call.
val callCapabilities = CallAttributesCompat.CallCapability(
supportsSetInactive = CallAttributesCompat.SUPPORTS_SET_INACTIVE // Call can be made inactive (implies hold)
)
return CallAttributesCompat(
displayName = callerName,
address = addressUri,
direction = CallAttributesCompat.DIRECTION_INCOMING,
callType = if (isVideoCall) CallAttributesCompat.CALL_TYPE_VIDEO_CALL else CallAttributesCompat.CALL_TYPE_AUDIO_CALL,
callCapabilitiesCompat = callCapabilities
)
}
新增通話
使用 callsManager.addCall 和 CallAttributesCompat,以及回呼,將新通話新增至系統,並管理遠端介面更新。addCall 區塊中的 callControlScope 主要可讓應用程式轉換通話狀態及接收音訊更新:
try {
callsManager.addCall(
INCOMING_CALL_ATTRIBUTES,
onAnswerCall, // Watch needs to know if it can answer the call.
onSetCallDisconnected,
onSetCallActive,
onSetCallInactive
) {
// The call was successfully added once this scope runs.
callControlScope = this
}
}
catch(addCallException: Exception){
// Handle the addCall failure.
}
接聽來電
在 CallControlScope 內接聽來電:
when (val result = answer(CallAttributesCompat.CALL_TYPE_AUDIO_CALL)) {
is CallControlResult.Success -> { /* Call answered */ }
is CallControlResult.Error -> { /* Handle error */ }
}
拒接來電
使用 disconnect() 和 DisconnectCause.REJECTED 在 CallControlScope 中拒接來電:
disconnect(DisconnectCause(DisconnectCause.REJECTED))
撥出電話
在遠端接聽電話後,將撥出電話設為啟用:
when (val result = setActive()) {
is CallControlResult.Success -> { /* Call active */ }
is CallControlResult.Error -> { /* Handle error */ }
}
保留通話
使用 setInactive() 保留通話:
when (val result = setInactive()) {
is CallControlResult.Success -> { /* Call on hold */ }
is CallControlResult.Error -> { /* Handle error */ }
}
中斷通話
使用 disconnect() 透過 DisconnectCause 掛斷電話:
disconnect(DisconnectCause(DisconnectCause.LOCAL))
管理通話音訊端點
使用 currentCallEndpoint、availableEndpoints 和 isMuted Flow 觀察及管理 CallControlScope 內的音訊端點
fun observeAudioStateChanges(callControlScope: CallControlScope) {
with(callControlScope) {
launch { currentCallEndpoint.collect { /* Update UI */ } }
launch { availableEndpoints.collect { /* Update UI */ } }
launch { isMuted.collect { /* Handle mute state */ } }
}
}
使用 requestEndpointChange() 變更使用的音訊裝置:
coroutineScope.launch {
callControlScope.requestEndpointChange(callEndpoint)
}
前景支援
這個程式庫會使用 ConnectionService (Android 13 API 級別 33 以下版本) 或foregroundtypes (Android 14 API 級別 34 以上版本) 支援前景作業。
根據前景服務規定,應用程式必須發布通知,讓使用者瞭解應用程式正在前景執行。
為確保應用程式取得前景執行優先權,請在新增與平台的通話後建立通知。應用程式終止通話或通知失效時,系統會移除前景優先順序。
遠端 Surface 支援
遠端裝置 (智慧手錶、藍牙耳機、Android Auto) 能夠管理通話,不必直接操作手機。應用程式必須實作提供給 CallsManager.addCall 的回呼 Lambda (onAnswerCall、onSetCallDisconnected、onSetCallActive、onSetCallInactive),以處理這些裝置啟動的動作。
發生遠端動作時,系統會叫用對應的 lambda。
Lambda 成功完成,表示指令已處理完畢。如果無法遵守指令,Lambda 應擲回例外狀況。
正確實作可確保不同裝置間的通話控制功能運作無礙。 使用各種遙控器表面徹底測試。
來電額外資訊
除了管理通話狀態和音訊路徑外,這個程式庫也支援通話擴充功能。應用程式可實作這些選用功能,在 Android Auto 等遠端途徑上提供更豐富的通話體驗。這些功能包括會議室、通話靜音和額外的通話圖示。應用程式實作擴充功能時,應用程式提供的資訊會與所有連線裝置同步,這些裝置也支援在 UI 中顯示這些擴充功能。也就是說,使用者也能在遠端裝置上使用這些功能。
建立含有擴充功能的通話
建立通話時,您可以改用 CallManager#addCallWithExtensions,而非使用 CallManager#addCall 建立通話,這樣應用程式就能存取名為 ExtensionInitializationScope 的不同範圍。這個範圍可讓應用程式初始化支援的一組選用擴充功能。此外,這個範圍還提供額外方法 onCall,可在擴充功能能力交換和初始化完成後,將 CallControlScope 提供給應用程式。
scope.launch {
mCallsManager.addCallWithExtensions(
attributes,
onAnswer,
onDisconnect,
onSetActive,
onSetInactive
) {
// Initialize extension-specific code...
// After the call has been initialized, perform in-call actions
onCall {
// Example: process call state updates
callStateFlow.onEach { newState ->
// handle call state updates and notify telecom
}.launchIn(this)
// Use initialized extensions...
}
}
}
支援通話參與者
如果應用程式支援會議或群組通話的通話參與者,請使用 addParticipantExtension 宣告支援這項擴充功能,並在參與者變更時使用相關 API 更新遠端介面。
mCallsManager.addCallWithExtensions(...) {
// Initialize extensions...
// Notifies Jetpack that this app supports the participant
// extension and provides the initial participants state in the call.
val participantExtension = addParticipantExtension(
initialParticipants,
initialActiveParticipant
)
// After the call has been initialized, perform in-call control actions
onCall {
// other in-call control and extension actions...
// Example: update remote surfaces when the call participants change
participantsFlow.onEach { newParticipants ->
participantExtension.updateParticipants(newParticipants)
}.launchIn(this)
}
}
除了通知遠端介面通話參與者,您也可以使用 ParticipantExtension#updateActiveParticipant 更新活躍參與者。
此外,系統也支援與通話參與者相關的選用動作。
應用程式可以使用 ParticipantExtension#addRaiseHandSupport 支援參與者在通話中舉手的概念,並查看其他哪些參與者也舉手。
mCallsManager.addCallWithExtensions(...) {
// Initialize extensions...
// Notifies Jetpack that this app supports the participant
// extension and provides the initial list of participants in the call.
val participantExtension = addParticipantExtension(initialParticipants)
// Notifies Jetpack that this app supports the notion of participants
// being able to raise and lower their hands.
val raiseHandState = participantExtension.addRaiseHandSupport(
initialRaisedHands
) { onHandRaisedStateChanged ->
// handle this user's raised hand state changed updates from
// remote surfaces.
}
// After the call has been initialized, perform in-call control actions
onCall {
// other in-call control and extension actions...
// Example: update remote surfaces when the call participants change
participantsFlow.onEach { newParticipants ->
participantExtension.updateParticipants(newParticipants)
}.launchIn(this)
// notify remote surfaces of which of the participants have their
// hands raised
raisedHandsFlow.onEach { newRaisedHands ->
raiseHandState.updateRaisedHands(newRaisedHands)
}.launchIn(this)
}
}
支援通話靜音
通話靜音功能可讓使用者要求應用程式將通話的輸出音訊設為靜音,而不必實際將裝置麥克風設為靜音。這項功能是依據每次通話管理,因此在 VoIP 通話啟用時,Jetpack 會處理管理進行中行動網路通話全域靜音狀態的複雜性。這項做法可減少多方通話時的錯誤,並在使用者啟用通話靜音但未察覺時,提供「你正在說話」等實用提示。
mCallsManager.addCallWithExtensions(...) {
// Initialize extensions...
// Add support for locally silencing the call's outgoing audio and
// register a handler for when the user changes the call silence state
// from a remote surface.
val callSilenceExtension = addLocalCallSilenceExtension(
initialCallSilenceState = false
) { newCallSilenceStateRequest ->
// handle the user's request to enable/disable call silence from
// a remote surface
}
// After the call has been initialized, perform in-call control actions
onCall {
// other in-call control and extension actions...
// When the call's call silence state changes, update remote
// surfaces of the new state.
callSilenceState.onEach { isSilenced ->
callSilenceExtension.updateIsLocallySilenced(isSilenced)
}.launchIn(this)
}
}
支援通話圖示
應用程式可透過通話圖示指定代表通話的自訂圖示,在通話期間顯示於遠端 Surface。通話期間,這個圖示也可能會更新。
mCallsManager.addCallWithExtensions(...) {
// Initialize extensions...
// Add support for a custom call icon to be displayed during the
// lifetime of the call.
val callIconExtension = addCallIconExtension(
initialCallIconUri = initialUri
)
// After the call has been initialized, perform in-call control actions
onCall {
// other in-call control and extension actions...
// When the call's icon changes, update remote surfaces by providing
// the new URI.
callIconUri.onEach { newIconUri ->
callIconExtension.updateCallIconUri(newIconUri)
}.launchIn(this)
}
}
與系統通話記錄整合
應用程式可透過 Core-Telecom 整合系統通話記錄。這項功能可讓使用者透過系統撥號應用程式的通話記錄,查看使用 VoIP 應用程式撥打或接聽的電話,並透過撥號應用程式發起回撥。
啟用通話記錄整合功能
如要啟用這項功能,應用程式必須在 AndroidManifest.xml 中註冊,以處理 TelecomManager.ACTION_CALL_BACK 意圖,並提供 Activity 來處理回呼。舉例來說,假設 VoipCallbackActivity 是處理回呼的活動,則資訊清單應如下所示:
在資訊清單中註冊意圖
<activity
android:name=".VoipCallbackActivity"
android:exported="true">
<intent-filter>
<action android:name="android.telecom.action.CALL_BACK" />
</intent-filter>
</activity>
處理回呼意圖
使用者從系統撥號器啟動回呼時,系統會啟動已註冊的活動,並執行 TelecomManager.ACTION_CALL_BACK 動作。意圖包含您最初使用 CallsManager 新增通話時,提供給應用程式的 TelecomManager.EXTRA_UUID。
CallControlScope 方法 getCallId 會傳回通話的專屬 ID。
您應在應用程式中儲存啟動通話所需的所有資訊 (例如群組通話的電話號碼清單),以便在觸發回呼時重建並再次撥打電話。
class VoipCallbackActivity : Activity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
if (intent?.action == TelecomManager.ACTION_CALL_BACK) {
val uuidString = intent.getStringExtra(TelecomManager.EXTRA_UUID)
if (uuidString != null) {
val uuid = UUID.fromString(uuidString)
// Retrieve your persisted call info using the UUID
val callData = CallRepository.getCallByUuid(uuid)
if (callData != null) {
// Place the call again using your app's logic
// ...
}
}
}
finish()
}
}
停用通話記錄整合功能
系統預設會將通話記錄到系統通話記錄。如要針對每次通話停用這項功能,請在建立 CallAttributesCompat 時,將 isLogged 參數設為 false:
val callAttributes = CallAttributesCompat(
displayName = callerName,
address = addressUri,
direction = CallAttributesCompat.DIRECTION_OUTGOING,
isLogExcluded = true // Opt-out of system call logging
)