Mengambil detail untuk kontak

Tutorial ini menunjukkan cara mengambil data detail untuk kontak, seperti alamat email, nomor telepon, dan sebagainya. Itulah detail yang dicari pengguna saat mereka mengambil kontak. Anda dapat memberikan semua detail kontak, atau hanya menampilkan detail jenis tertentu, seperti alamat email.

Langkah-langkah dalam pelajaran ini mengasumsikan bahwa Anda sudah memiliki baris ContactsContract.Contacts untuk kontak yang dipilih pengguna. Pelajaran Mengambil nama kontak menunjukkan cara mengambil daftar kontak.

Mengambil semua detail kontak

Untuk mengambil semua detail kontak, telusuri tabel ContactsContract.Data untuk setiap baris yang berisi LOOKUP_KEY kontak. Kolom ini tersedia di tabel ContactsContract.Data karena Penyedia Kontak membuat penggabungan implisit antara tabel ContactsContract.Contacts dan tabel ContactsContract.Data. Kolom LOOKUP_KEY dijelaskan secara lebih detail dalam pelajaran Mengambil nama kontak.

Catatan: Mengambil semua detail untuk sebuah kontak dapat menurunkan performa perangkat karena perangkat harus mengambil semua kolom dalam tabel ContactsContract.Data. Pertimbangkan dampak performa sebelum Anda menggunakan teknik ini.

Meminta izin

Untuk membaca dari Penyedia Kontak, aplikasi Anda harus memiliki izin READ_CONTACTS. Untuk meminta izin ini, tambahkan elemen turunan <manifest> berikut ke file manifes Anda:

    <uses-permission android:name="android.permission.READ_CONTACTS" />

Menyiapkan proyeksi

Tergantung jenis data yang dimuat oleh baris, data mungkin menggunakan sedikit atau banyak kolom. Selain itu, data berada di kolom yang berbeda tergantung pada jenis datanya. Guna memastikan Anda mendapatkan semua kolom yang memungkinkan untuk semua kemungkinan jenis data, Anda harus menambahkan semua nama kolom ke proyeksi Anda. Selalu ambil Data._ID jika Anda mengikat hasil Cursor ke ListView; jika tidak, binding tidak akan berfungsi. Ambil juga Data.MIMETYPE agar Anda dapat mengidentifikasi jenis data dari setiap baris yang Anda ambil. Contoh:

Kotlin

private val PROJECTION: Array<out String> = arrayOf(
        ContactsContract.Data._ID,
        ContactsContract.Data.MIMETYPE,
        ContactsContract.Data.DATA1,
        ContactsContract.Data.DATA2,
        ContactsContract.Data.DATA3,
        ContactsContract.Data.DATA4,
        ContactsContract.Data.DATA5,
        ContactsContract.Data.DATA6,
        ContactsContract.Data.DATA7,
        ContactsContract.Data.DATA8,
        ContactsContract.Data.DATA9,
        ContactsContract.Data.DATA10,
        ContactsContract.Data.DATA11,
        ContactsContract.Data.DATA12,
        ContactsContract.Data.DATA13,
        ContactsContract.Data.DATA14,
        ContactsContract.Data.DATA15
)

Java

    private static final String[] PROJECTION =
            {
                ContactsContract.Data._ID,
                ContactsContract.Data.MIMETYPE,
                ContactsContract.Data.DATA1,
                ContactsContract.Data.DATA2,
                ContactsContract.Data.DATA3,
                ContactsContract.Data.DATA4,
                ContactsContract.Data.DATA5,
                ContactsContract.Data.DATA6,
                ContactsContract.Data.DATA7,
                ContactsContract.Data.DATA8,
                ContactsContract.Data.DATA9,
                ContactsContract.Data.DATA10,
                ContactsContract.Data.DATA11,
                ContactsContract.Data.DATA12,
                ContactsContract.Data.DATA13,
                ContactsContract.Data.DATA14,
                ContactsContract.Data.DATA15
            };

Proyeksi ini mengambil semua kolom untuk sebuah baris dalam tabel ContactsContract.Data, menggunakan nama kolom yang ditentukan dalam class ContactsContract.Data.

Jika ingin, Anda juga dapat menggunakan konstanta kolom lain yang ditentukan di atau diwarisi oleh class ContactsContract.Data. Namun, perhatikan bahwa kolom SYNC1 hingga SYNC4 dimaksudkan untuk digunakan oleh adaptor sinkronisasi, sehingga datanya tidak berguna.

Menentukan kriteria pemilihan

Tentukan konstanta untuk klausa pemilihan, array untuk menampung argumen pemilihan, dan variabel untuk menyimpan nilai pemilihan. Gunakan kolom Contacts.LOOKUP_KEY untuk menemukan kontak. Contoh:

Kotlin

// Defines the selection clause
private const val SELECTION: String = "${ContactsContract.Data.LOOKUP_KEY} = ?"
...
// Defines the array to hold the search criteria
private val selectionArgs: Array<String> = arrayOf("")
/*
 * Defines a variable to contain the selection value. Once you
 * have the Cursor from the Contacts table, and you've selected
 * the desired row, move the row's LOOKUP_KEY value into this
 * variable.
 */
private var lookupKey: String? = null

Java

    // Defines the selection clause
    private static final String SELECTION = Data.LOOKUP_KEY + " = ?";
    // Defines the array to hold the search criteria
    private String[] selectionArgs = { "" };
    /*
     * Defines a variable to contain the selection value. Once you
     * have the Cursor from the Contacts table, and you've selected
     * the desired row, move the row's LOOKUP_KEY value into this
     * variable.
     */
    private lateinit var lookupKey: String

Menggunakan "?" sebagai placeholder dalam ekspresi teks pilihan Anda memastikan bahwa hasil penelusuran dihasilkan melalui binding, bukan kompilasi SQL. Pendekatan ini menghilangkan kemungkinan injeksi SQL yang berbahaya.

Menentukan urutan penyortiran

Tetapkan urutan penyortiran yang Anda inginkan dalam Cursor yang dihasilkan. Untuk menyatukan semua baris untuk jenis data tertentu, urutkan berdasarkan Data.MIMETYPE. Argumen kueri ini mengelompokkan semua baris email menjadi satu, semua baris telepon menjadi satu, dan seterusnya. Contoh:

Kotlin

/*
 * Defines a string that specifies a sort order of MIME type
 */
private const val SORT_ORDER = ContactsContract.Data.MIMETYPE

Java

    /*
     * Defines a string that specifies a sort order of MIME type
     */
    private static final String SORT_ORDER = ContactsContract.Data.MIMETYPE;

Catatan: Beberapa jenis data tidak menggunakan subjenis, jadi Anda tidak dapat menyortir menurut subjenis. Sebagai gantinya, Anda harus melakukan iterasi melalui Cursor yang ditampilkan, menentukan jenis data baris saat ini, dan menyimpan data untuk baris yang menggunakan subjenis. Setelah selesai membaca kursor, Anda dapat mengurutkan setiap jenis data menurut subjenis dan menampilkan hasilnya.

Menginisialisasi loader

Selalu lakukan pengambilan data dari Penyedia Kontak (dan semua penyedia konten lainnya) di thread latar belakang. Gunakan framework Loader yang ditentukan oleh class LoaderManager dan antarmuka LoaderManager.LoaderCallbacks untuk melakukan pengambilan di latar belakang.

Setelah siap mengambil baris, inisialisasi framework loader dengan memanggil initLoader(). Teruskan ID bilangan bulat ke metode; ID ini akan diteruskan ke metode LoaderManager.LoaderCallbacks. Dengan ID ini, Anda dapat menggunakan beberapa loader pada aplikasi sehingga Anda dapat membedakannya.

Cuplikan berikut menunjukkan cara menginisialisasi framework loader:

Kotlin

// Defines a constant that identifies the loader
private const val DETAILS_QUERY_ID: Int = 0

class DetailsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor> {
    ...
    override fun onCreate(savedInstanceState: Bundle?) {
        ...
        // Initializes the loader framework
        loaderManager.initLoader(DETAILS_QUERY_ID, null, this)

Java

public class DetailsFragment extends Fragment implements
        LoaderManager.LoaderCallbacks<Cursor> {
    ...
    // Defines a constant that identifies the loader
    static int DETAILS_QUERY_ID = 0;
    ...
    @Override
    public void onCreate(Bundle savedInstanceState) {
        ...
        // Initializes the loader framework
        getLoaderManager().initLoader(DETAILS_QUERY_ID, null, this);

Mengimplementasikan onCreateLoader()

Implementasikan metode onCreateLoader(), yang dipanggil oleh framework loader segera setelah Anda memanggil initLoader(). Tampilkan CursorLoader dari metode ini. Karena Anda menelusuri tabel ContactsContract.Data, gunakan konstanta Data.CONTENT_URI sebagai URI konten. Contoh:

Kotlin

override fun onCreateLoader(loaderId: Int, args: Bundle?): Loader<Cursor> {
    // Choose the proper action
    mLoader = when(loaderId) {
        DETAILS_QUERY_ID -> {
            // Assigns the selection parameter
            selectionArgs[0] = lookupKey
            // Starts the query
            activity?.let {
                CursorLoader(
                        it,
                        ContactsContract.Data.CONTENT_URI,
                        PROJECTION,
                        SELECTION,
                        selectionArgs,
                        SORT_ORDER
                )
            }
        }
        ...
    }
    return mLoader
}

Java

@Override
    public Loader<Cursor> onCreateLoader(int loaderId, Bundle args) {
        // Choose the proper action
        switch (loaderId) {
            case DETAILS_QUERY_ID:
            // Assigns the selection parameter
            selectionArgs[0] = lookupKey;
            // Starts the query
            CursorLoader mLoader =
                    new CursorLoader(
                            getActivity(),
                            ContactsContract.Data.CONTENT_URI,
                            PROJECTION,
                            SELECTION,
                            selectionArgs,
                            SORT_ORDER
                    );
    }

Mengimplementasikan onLoadFinished() dan onLoaderReset()

Implementasikan metode onLoadFinished(). Framework loader ini memanggil onLoadFinished() saat Penyedia Kontak menampilkan hasil kueri. Contoh:

Kotlin

    override fun onLoadFinished(loader: Loader<Cursor>, data: Cursor) {
        when(loader.id) {
            DETAILS_QUERY_ID -> {
                /*
                 * Process the resulting Cursor here.
                 */
            }
            ...
        }
    }

Java

    @Override
    public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
        switch (loader.getId()) {
            case DETAILS_QUERY_ID:
                    /*
                     * Process the resulting Cursor here.
                     */
                }
                break;
            ...
        }
    }

Metode onLoaderReset() dipanggil saat framework loader mendeteksi bahwa data yang mendukung hasil Cursor telah berubah. Pada tahap ini, hapus semua referensi yang ada ke Cursor dengan menetapkannya ke null. Jika tidak, framework loader tidak akan menghancurkan Cursor lama, dan kebocoran memori akan terjadi. Contoh:

Kotlin

    override fun onLoaderReset(loader: Loader<Cursor>) {
        when (loader.id) {
            DETAILS_QUERY_ID -> {
                /*
                 * If you have current references to the Cursor,
                 * remove them here.
                 */
            }
            ...
        }
    }

Java

    @Override
    public void onLoaderReset(Loader<Cursor> loader) {
        switch (loader.getId()) {
            case DETAILS_QUERY_ID:
                /*
                 * If you have current references to the Cursor,
                 * remove them here.
                 */
                }
                break;
    }

Mengambil detail spesifik untuk sebuah kontak

Pengambilan jenis data tertentu untuk suatu kontak, misalnya semua email, mengikuti pola yang sama dengan mengambil semua detail. Inilah satu-satunya perubahan yang perlu Anda buat pada kode yang tercantum di Mengambil semua detail untuk kontak:

Proyeksi
Ubah proyeksi Anda untuk mengambil kolom-kolom yang spesifik untuk jenis data tersebut. Ubah juga proyeksi untuk menggunakan konstanta nama kolom yang ditentukan dalam subclass ContactsContract.CommonDataKinds yang sesuai dengan jenis data.
Pilihan
Ubah teks pemilihan untuk menelusuri nilai MIMETYPE yang spesifik untuk jenis data Anda.
Urutan penyortiran
Karena Anda hanya memilih satu jenis detail, jangan mengelompokkan Cursor yang ditampilkan menurut Data.MIMETYPE.

Modifikasi ini dijelaskan di bagian berikut.

Menentukan proyeksi

Tentukan kolom yang ingin Anda ambil, menggunakan konstanta nama kolom di subclass ContactsContract.CommonDataKinds untuk jenis data. Jika Anda berencana mengikat Cursor dengan ListView, pastikan untuk mengambil kolom _ID. Misalnya, untuk mengambil data email, tentukan proyeksi berikut:

Kotlin

private val PROJECTION: Array<String> = arrayOf(
        ContactsContract.CommonDataKinds.Email._ID,
        ContactsContract.CommonDataKinds.Email.ADDRESS,
        ContactsContract.CommonDataKinds.Email.TYPE,
        ContactsContract.CommonDataKinds.Email.LABEL
)

Java

    private static final String[] PROJECTION =
            {
                ContactsContract.CommonDataKinds.Email._ID,
                ContactsContract.CommonDataKinds.Email.ADDRESS,
                ContactsContract.CommonDataKinds.Email.TYPE,
                ContactsContract.CommonDataKinds.Email.LABEL
            };

Perhatikan bahwa proyeksi ini menggunakan nama kolom yang ditentukan dalam class ContactsContract.CommonDataKinds.Email, bukan nama kolom yang ditentukan dalam class ContactsContract.Data. Penggunaan nama kolom khusus email membuat kode lebih mudah dibaca.

Dalam proyeksi ini, Anda juga dapat menggunakan salah satu kolom lain yang ditentukan dalam subclass ContactsContract.CommonDataKinds.

Menentukan kriteria pemilihan

Tentukan ekspresi teks penelusuran yang mengambil baris untuk LOOKUP_KEY kontak tertentu dan Data.MIMETYPE detail yang Anda inginkan. Sertakan nilai MIMETYPE dalam tanda kutip tunggal dengan menyambungkan karakter "'" (tanda kutip tunggal) ke awal dan akhir konstanta; jika tidak, penyedia akan menafsirkan konstanta tersebut sebagai nama variabel, bukan sebagai nilai string. Anda tidak perlu menggunakan placeholder untuk nilai ini, karena Anda menggunakan konstanta, bukan nilai yang disediakan pengguna. Contoh:

Kotlin

/*
 * Defines the selection clause. Search for a lookup key
 * and the Email MIME type
 */
private const val SELECTION =
        "${ContactsContract.Data.LOOKUP_KEY} = ? AND " +
        "${ContactsContract.Data.MIMETYPE} = '${Email.CONTENT_ITEM_TYPE}'"
...
// Defines the array to hold the search criteria
private val selectionArgs: Array<String> = arrayOf("")

Java

    /*
     * Defines the selection clause. Search for a lookup key
     * and the Email MIME type
     */
    private static final String SELECTION =
            Data.LOOKUP_KEY + " = ?" +
            " AND " +
            Data.MIMETYPE + " = " +
            "'" + Email.CONTENT_ITEM_TYPE + "'";
    // Defines the array to hold the search criteria
    private String[] selectionArgs = { "" };

Menentukan urutan penyortiran

Tetapkan urutan penyortiran untuk Cursor yang ditampilkan. Karena Anda mengambil jenis data tertentu, hapus pengurutan berdasarkan MIMETYPE. Sebagai gantinya, jika jenis data detail yang Anda telusuri menyertakan subjenis, lakukan penyortiran menurut subjenis tersebut. Misalnya, untuk data email Anda dapat mengurutkan berdasarkan Email.TYPE:

Kotlin

private const val SORT_ORDER: String = "${Email.TYPE} ASC"

Java

    private static final String SORT_ORDER = Email.TYPE + " ASC ";