Thư viện Core-Telecom
đơn giản hoá quy trình tích hợp ứng dụng gọi với nền tảng Android bằng cách cung cấp một bộ API mạnh mẽ và nhất quán
Nếu muốn tìm hiểu cách triển khai thực tế, bạn có thể tìm thấy các ứng dụng mẫu trên GitHub:
- Ứng dụng mẫu gọn nhẹ – Một ví dụ tối thiểu minh hoạ cách sử dụng API
Core-Telecom
. Lý tưởng để nhanh chóng hiểu các khái niệm cơ bản. - Ứng dụng mẫu toàn diện (do Nhóm Core-Telecom phát triển) – Một ứng dụng có nhiều tính năng hơn, giới thiệu các chức năng Viễn thông nâng cao và các phương pháp hay nhất. Đây là một tài nguyên tuyệt vời để hiểu các tình huống tích hợp phức tạp.
Thiết lập Core-Telecom
Thêm phần phụ thuộc androidx.core:core-telecom
vào tệp build.gradle
của ứng dụng:
dependencies {
implementation ("androidx.core:core-telecom:1.0.0")
}
Khai báo quyền MANAGE_OWN_CALLS
trong AndroidManifest.xml
:
<uses-permission android:name="android.permission.MANAGE_OWN_CALLS" />
Đăng ký ứng dụng của bạn
Đăng ký ứng dụng gọi điện của bạn với Android bằng CallsManager
để bắt đầu thêm lệnh gọi vào hệ thống. Khi đăng ký, hãy chỉ định các chức năng của ứng dụng (ví dụ: hỗ trợ âm thanh, video):
val callsManager = CallsManager(context)
val capabilities: @CallsManager.Companion.Capability Int =
(CallsManager.CAPABILITY_BASELINE or
CallsManager.CAPABILITY_SUPPORTS_VIDEO_CALLING)
callsManager.registerAppWithTelecom(capabilities)
Quản lý cuộc gọi
Sử dụng API Core-Telecom để tạo và quản lý vòng đời cuộc gọi.
Tạo lệnh gọi
Đối tượng CallAttributesCompat
xác định các thuộc tính của một lệnh gọi duy nhất, lệnh gọi này có thể có các đặc điểm sau:
displayName
: tên người gọi.address
: Địa chỉ cuộc gọi (ví dụ: số điện thoại, đường liên kết đến cuộc họp).direction
: Đến hoặc đi.callType
: Âm thanh hoặc video.callCapabilities
: Hỗ trợ chuyển và giữ.
Sau đây là ví dụ về cách tạo cuộc gọi đến:
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
)
}
Thêm cuộc gọi
Sử dụng callsManager.addCall
với CallAttributesCompat
và lệnh gọi lại để thêm một lệnh gọi mới vào hệ thống và quản lý các bản cập nhật bề mặt từ xa. callControlScope
trong khối addCall
chủ yếu cho phép ứng dụng chuyển đổi trạng thái cuộc gọi và nhận thông tin cập nhật về âm thanh:
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.
}
Trả lời cuộc gọi
Trả lời cuộc gọi đến trong CallControlScope
:
when (val result = answer(CallAttributesCompat.CALL_TYPE_AUDIO_CALL)) {
is CallControlResult.Success -> { /* Call answered */ }
is CallControlResult.Error -> { /* Handle error */ }
}
Từ chối cuộc gọi
Từ chối cuộc gọi bằng cách sử dụng disconnect()
với DisconnectCause.REJECTED
trong CallControlScope
:
disconnect(DisconnectCause(DisconnectCause.REJECTED))
Đang thực hiện cuộc gọi đi
Đặt cuộc gọi đi thành đang hoạt động sau khi bên từ xa trả lời:
when (val result = setActive()) {
is CallControlResult.Success -> { /* Call active */ }
is CallControlResult.Error -> { /* Handle error */ }
}
Giữ cuộc gọi
Sử dụng setInactive()
để giữ cuộc gọi:
when (val result = setInactive()) {
is CallControlResult.Success -> { /* Call on hold */ }
is CallControlResult.Error -> { /* Handle error */ }
}
Ngắt kết nối cuộc gọi
Ngắt kết nối cuộc gọi bằng disconnect()
với DisconnectCause
:
disconnect(DisconnectCause(DisconnectCause.LOCAL))
Quản lý điểm cuối âm thanh cuộc gọi
Quan sát và quản lý các điểm cuối âm thanh bằng cách sử dụng Flow
currentCallEndpoint
, availableEndpoints
và isMuted
trong CallControlScope
fun observeAudioStateChanges(callControlScope: CallControlScope) {
with(callControlScope) {
launch { currentCallEndpoint.collect { /* Update UI */ } }
launch { availableEndpoints.collect { /* Update UI */ } }
launch { isMuted.collect { /* Handle mute state */ } }
}
}
Thay đổi thiết bị âm thanh đang hoạt động bằng requestEndpointChange()
:
coroutineScope.launch {
callControlScope.requestEndpointChange(callEndpoint)
}
Hỗ trợ trên nền trước
Thư viện này sử dụng ConnectionService
(Android 13 API cấp 33 trở xuống) hoặc foregroundtypes (Android 14 API cấp 34 trở lên) để hỗ trợ chế độ nền trước.
Theo yêu cầu về chế độ nền trước, ứng dụng phải đăng thông báo để người dùng biết rằng ứng dụng đang chạy ở chế độ nền trước.
Để đảm bảo ứng dụng của bạn được ưu tiên thực thi trên nền trước, hãy tạo một thông báo sau khi bạn thêm lệnh gọi với nền tảng. Mức độ ưu tiên ở nền trước sẽ bị xoá khi ứng dụng của bạn chấm dứt lệnh gọi hoặc thông báo của bạn không còn hợp lệ.
Tìm hiểu thêm về dịch vụ trên nền trước.
Hỗ trợ Surface từ xa
Các thiết bị từ xa (đồng hồ thông minh, tai nghe Bluetooth, Android Auto) có thể quản lý cuộc gọi mà không cần tương tác trực tiếp với điện thoại. Ứng dụng của bạn phải triển khai các hàm callback lambda (onAnswerCall
, onSetCallDisconnected
, onSetCallActive
, onSetCallInactive
) được cung cấp cho CallsManager.addCall
để xử lý các thao tác do các thiết bị này khởi tạo.
Khi một thao tác từ xa xảy ra, lambda tương ứng sẽ được gọi.
Hoàn tất thành công các tín hiệu lambda cho biết lệnh đã được xử lý. Nếu không thể tuân theo lệnh, hàm lambda sẽ gửi một ngoại lệ.
Việc triển khai đúng cách đảm bảo tính năng điều khiển cuộc gọi liền mạch trên nhiều thiết bị. Kiểm thử kỹ lưỡng với nhiều bề mặt từ xa.
Phần mở rộng về cuộc gọi
Ngoài việc quản lý trạng thái cuộc gọi và tuyến âm thanh của cuộc gọi, thư viện này cũng hỗ trợ các tiện ích cuộc gọi. Đây là các tính năng không bắt buộc mà ứng dụng của bạn có thể triển khai để mang lại trải nghiệm gọi phong phú hơn trên các nền tảng từ xa, chẳng hạn như Android Auto. Các tính năng này bao gồm phòng họp, tắt tiếng cuộc gọi và các biểu tượng cuộc gọi khác. Khi ứng dụng của bạn triển khai một tiện ích, thông tin mà ứng dụng cung cấp sẽ được đồng bộ hoá với tất cả các thiết bị đã kết nối cũng hỗ trợ hiển thị các tiện ích này trong giao diện người dùng. Điều này có nghĩa là người dùng cũng có thể tương tác với các tính năng này trên các thiết bị từ xa.
Tạo lệnh gọi có phần mở rộng
Khi tạo cuộc gọi, thay vì sử dụng CallManager#addCall
để tạo cuộc gọi, bạn có thể sử dụng CallManager#addCallWithExtensions. Thao tác này sẽ cấp cho ứng dụng quyền truy cập vào một phạm vi khác có tên là ExtensionInitializationScope
. Phạm vi này cho phép ứng dụng khởi chạy tập hợp các tiện ích không bắt buộc mà ứng dụng hỗ trợ. Ngoài ra, phạm vi này cung cấp thêm một phương thức là onCall
. Phương thức này sẽ cung cấp CallControlScope
trở lại cho ứng dụng sau khi quá trình trao đổi và khởi chạy chức năng mở rộng hoàn tất.
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...
}
}
}
Người tham gia cuộc gọi hỗ trợ
Nếu ứng dụng của bạn hỗ trợ người tham gia cuộc gọi cho các cuộc họp hoặc cuộc gọi nhóm, hãy sử dụng addParticipantExtension
để khai báo hỗ trợ tiện ích này và sử dụng các API liên quan để cập nhật các nền tảng từ xa khi người tham gia thay đổi.
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)
}
}
Cùng với việc thông báo cho các nền tảng từ xa về những người tham gia cuộc gọi, bạn cũng có thể cập nhật người tham gia đang hoạt động bằng cách sử dụng ParticipantExtension#updateActiveParticipant
.
Ngoài ra, bạn cũng có thể thực hiện các thao tác không bắt buộc liên quan đến người tham gia cuộc gọi.
Ứng dụng có thể sử dụng ParticipantExtension#addRaiseHandSupport
để hỗ trợ khái niệm người tham gia giơ tay trong cuộc gọi và xem những người tham gia khác cũng giơ tay.
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)
}
}
Hỗ trợ tính năng Tắt tiếng cuộc gọi
Tính năng tắt tiếng cuộc gọi cho phép người dùng yêu cầu ứng dụng tắt tiếng âm thanh đi của cuộc gọi mà không cần tắt tiếng micrô của thiết bị. Tính năng này được quản lý theo từng lệnh gọi, vì vậy, Jetpack sẽ xử lý sự phức tạp trong việc quản lý trạng thái tắt tiếng chung của các cuộc gọi di động đang diễn ra trong khi cuộc gọi VOIP đang hoạt động. Điều này giúp giảm lỗi khi tắt tiếng âm thanh phát đi trong các trường hợp nhiều cuộc gọi, đồng thời cho phép sử dụng các tính năng hữu ích như chỉ báo "bạn đang nói" khi người dùng đang nói mà không nhận ra rằng tính năng tắt tiếng cuộc gọi đang bật.
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)
}
}
Biểu tượng cuộc gọi hỗ trợ
Biểu tượng cuộc gọi cho phép ứng dụng chỉ định một biểu tượng tuỳ chỉnh đại diện cho cuộc gọi sẽ xuất hiện trên các nền tảng từ xa trong cuộc gọi. Biểu tượng này cũng có thể được cập nhật trong suốt thời gian của lệnh gọi.
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)
}
}