Интерфейс выбора контактов Android Contact Picker — это стандартизированный, удобный для просмотра интерфейс, позволяющий пользователям делиться контактами с вашим приложением. Доступный на устройствах под управлением Android 17 (уровень API 37) или выше, этот интерфейс предлагает альтернативу широкому разрешению READ_CONTACTS , обеспечивающую конфиденциальность. Вместо запроса доступа ко всей адресной книге пользователя, ваше приложение указывает необходимые поля данных, такие как номера телефонов или адреса электронной почты, а пользователь выбирает конкретные контакты для обмена. Это предоставляет вашему приложению доступ на чтение только к выбранным данным, обеспечивая детальный контроль и предоставляя единообразный пользовательский опыт со встроенным поиском, переключением профилей и возможностью множественного выбора без необходимости разработки или поддержки пользовательского интерфейса.
Интегрируйте средство выбора контактов.
Для интеграции средства выбора контактов используйте интент Intent.ACTION_PICK_CONTACTS . Этот интент запускает средство выбора и возвращает выбранные контакты в ваше приложение.
В отличие от устаревшего ACTION_PICK , Contact Picker позволяет одновременно указывать несколько полей данных, необходимых вашему приложению. Это делается с помощью Intent.EXTRA_REQUESTED_DATA_FIELDS , передавая ArrayList<String> MIME-типов, определенных в ContactsContract.CommonDataKinds .
К распространённым MIME-типам относятся:
-
ContactsContract.CommonDataKinds.Phone.CONTENT_ITEM_TYPE -
ContactsContract.CommonDataKinds.Email.CONTENT_ITEM_TYPE -
ContactsContract.CommonDataKinds.StructuredPostal.CONTENT_ITEM_TYPE
Запустите средство выбора
Используйте registerForActivityResult с контрактом StartActivityForResult для запуска средства выбора. Вы можете настроить Intent таким образом, чтобы разрешить одиночный или множественный выбор.
// Launcher for the Contact Picker intent
val pickContact = rememberLauncherForActivityResult(StartActivityForResult()) {
if (it.resultCode == Activity.RESULT_OK) {
val resultUri = it.data?.data ?: return@rememberLauncherForActivityResult
// Process the result URI in a background thread to fetch all selected contacts
coroutine.launch {
contacts = processContactPickerResultUri(resultUri, context)
}
}
}
Режим выбора
Интерфейс средства выбора контактов подстраивается под запрашиваемые поля данных. В зависимости от этих требований пользователи могут либо выбрать всю запись контакта целиком, если требуется несколько полей, либо выбрать конкретные элементы данных из информации о контакте.

Выберите один контакт
В этом примере приложение запрашивает только номера телефонов. Средство выбора отфильтрует список, чтобы отобразить только контакты с номерами телефонов, и позволит пользователю выбрать конкретный номер.
// Define the specific contact data fields you need
val requestedFields = arrayListOf(
Email.CONTENT_ITEM_TYPE,
Phone.CONTENT_ITEM_TYPE,
)
// Set up the intent for the Contact Picker
val pickContactIntent = Intent(ACTION_PICK_CONTACTS).apply {
putStringArrayListExtra(
EXTRA_PICK_CONTACTS_REQUESTED_DATA_FIELDS,
requestedFields
)
}
// Launch the picker
pickContact.launch(pickContactIntent)
Выберите несколько контактов
Чтобы включить множественный выбор, добавьте дополнительный параметр Intent.EXTRA_ALLOW_MULTIPLE . При желании вы можете ограничить количество элементов, которые может выбрать пользователь.
val requestedFields = arrayListOf(
Email.CONTENT_ITEM_TYPE,
Phone.CONTENT_ITEM_TYPE,
)
// Set up the intent for the Contact Picker
val pickContactIntent = Intent(ACTION_PICK_CONTACTS).apply {
// Enable multi-select
putExtra(Intent.EXTRA_ALLOW_MULTIPLE, true)
// Set limit of selectable contacts
putExtra(EXTRA_PICK_CONTACTS_SELECTION_LIMIT, 5)
// Define the specific contact data fields you need
putStringArrayListExtra(
EXTRA_PICK_CONTACTS_REQUESTED_DATA_FIELDS,
requestedFields
)
// Enable this option to only filter contacts that have all the requested data fields
putExtra(EXTRA_PICK_CONTACTS_MATCH_ALL_DATA_FIELDS, false)
}
// Launch the picker
pickContact.launch(pickContactIntent)
Обработайте результаты
После завершения выбора пользователем система возвращает объект RESULT_OK и URI сессии. Этот URI предоставляет временный доступ для чтения к выбранным данным.
Вы можете запросить этот URI, используя стандартный ContentResolver . Полученный Cursor содержит запрошенные поля данных и соответствует схеме ContactsContract.Data .
// Data class representing a parsed Contact with selected details.
data class Contact(
val lookupKey: String,
val name: String,
val emails: List<String>,
val phones: List<String>
)
// Helper function to query the content resolver with the URI returned by the Contact Picker.
// Parses the cursor to extract contact details such as name, email, and phone number.
private suspend fun processContactPickerResultUri(
sessionUri: Uri,
context: Context
): List<Contact> = withContext(Dispatchers.IO) {
// Define the columns we want to retrieve from the ContactPicker ContentProvider
val projection = arrayOf(
ContactsContract.Contacts.LOOKUP_KEY,
ContactsContract.Contacts.DISPLAY_NAME_PRIMARY,
ContactsContract.Data.MIMETYPE, // Type of data (e.g., email or phone)
ContactsContract.Data.DATA1, // The actual data (Phone number / Email string)
)
// We use `LOOKUP_KEY` as a unique ID to aggregate all contact info related to a same person
val contactsMap = mutableMapOf<String, Contact>()
// Note: The Contact Picker Session Uri doesn't support custom selection & selectionArgs.
// We query the URI directly to get the results chosen by the user.
context.contentResolver.query(sessionUri, projection, null, null, null)?.use { cursor ->
// Get the column indices for our requested projection
val lookupKeyIdx = cursor.getColumnIndex(ContactsContract.Contacts.LOOKUP_KEY)
val mimeTypeIdx = cursor.getColumnIndex(ContactsContract.Data.MIMETYPE)
val nameIdx = cursor.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME_PRIMARY)
val data1Idx = cursor.getColumnIndex(ContactsContract.Data.DATA1)
while (cursor.moveToNext()) {
val lookupKey = cursor.getString(lookupKeyIdx)
val mimeType = cursor.getString(mimeTypeIdx)
val name = cursor.getString(nameIdx) ?: ""
val data1 = cursor.getString(data1Idx) ?: ""
val email = if (mimeType == Email.CONTENT_ITEM_TYPE) data1 else null
val phone = if (mimeType == Phone.CONTENT_ITEM_TYPE) data1 else null
val existingContact = contactsMap[lookupKey]
if (existingContact != null) {
contactsMap[lookupKey] = existingContact.copy(
emails = if (email != null) existingContact.emails + email else existingContact.emails,
phones = if (phone != null) existingContact.phones + phone else existingContact.phones
)
} else {
contactsMap[lookupKey] = Contact(
lookupKey = lookupKey,
name = name,
emails = if (email != null) listOf(email) else emptyList(),
phones = if (phone != null) listOf(phone) else emptyList()
)
}
}
}
return@withContext contactsMap.values.toList()
}
Обратная совместимость
Для приложений, ориентированных на Android 17 (уровень API 37) и выше, система автоматически обновляет существующий Intent.ACTION_PICK для использования нового интерфейса выбора контактов.
Если ваше приложение уже использует ACTION_PICK , вам не нужно менять код для получения нового пользовательского интерфейса. Однако, чтобы воспользоваться новыми функциями, такими как получение одного Uri для запроса контактных данных, переключение между личным и рабочим профилями или запросы нескольких полей данных, вам необходимо обновить свою реализацию, чтобы использовать Intent.ACTION_PICK_CONTACTS или новые дополнительные параметры Intent.
Тестирование на более старых целевых SDK.
Вы можете протестировать новое поведение средства выбора на устройствах под управлением Android 17 и выше, даже если ваше приложение ориентировано на более старую версию SDK, добавив логическую переменную EXTRA_USE_SYSTEM_CONTACTS_PICKER в ваш интент ACTION_PICK .
Передовые методы
- Запрашивайте только то, что вам нужно : если вашему приложению нужно только отправить SMS, запросите
Phone.CONTENT_ITEM_TYPE. Средство выбора автоматически отфильтрует контакты, у которых нет номеров телефонов, что обеспечит более удобный пользовательский интерфейс. - Управление несколькими записями данных для каждого контакта : Отдельные контакты часто содержат различные адреса электронной почты или номера телефонов. Для обеспечения четкого и интуитивно понятного отображения этой информации пользователю рекомендуется группировать контакты с помощью
ContactsContract.Contacts.LOOKUP_KEY. Кроме того, вы можете получить конкретные метки для каждой записи (например, рабочая или личная), чтобы предложить более детальные варианты выбора в интерфейсе вашего приложения. - Сохранение данных немедленно : URI сессии предоставляет временное разрешение на чтение. Если вам потребуется получить доступ к этой контактной информации позже (после завершения процесса вашего приложения), ваше приложение должно сохранить контактные данные.
- Не полагайтесь на данные учетной записи : для защиты конфиденциальности пользователей и предотвращения идентификации по отпечатку учетной записи метаданные, относящиеся к конкретной учетной записи, удаляются из результатов.