Cette leçon vous explique comment récupérer une liste de contacts dont les données correspondent à la totalité ou à une partie d'une chaîne de recherche, à l'aide des techniques suivantes:
- Correspondre aux noms des contacts
- Récupérez une liste de contacts en faisant correspondre la chaîne de recherche avec tout ou partie des données du nom du contact. Le fournisseur de contacts autorise plusieurs instances du même nom. Cette technique peut donc renvoyer une liste de correspondances.
- Correspondre à un type de données spécifique, comme un numéro de téléphone
- Récupérez une liste de contacts en faisant correspondre la chaîne de recherche à un type de données détaillées, tel qu'une adresse e-mail. Par exemple, cette technique vous permet de répertorier tous les contacts dont l'adresse e-mail correspond à la chaîne de recherche.
- Correspondre à tout type de données
- Récupérez une liste de contacts en faisant correspondre la chaîne de recherche à n'importe quel type de données détaillées, y compris le nom, le numéro de téléphone, l'adresse postale, l'adresse e-mail, etc. Par exemple, cette technique vous permet d'accepter n'importe quel type de données pour une chaîne de recherche, puis de lister les contacts pour lesquels les données correspondent à la chaîne.
Remarque:Tous les exemples de cette leçon utilisent un CursorLoader
pour récupérer des données auprès du fournisseur de contacts. Un CursorLoader
exécute sa requête sur un thread distinct du thread d'UI. Cela garantit que la requête ne ralentit pas les temps de réponse de l'UI et ne nuit pas à l'expérience utilisateur. Pour en savoir plus, consultez le cours de formation Android
Charger des données en arrière-plan.
Demander l'autorisation de lire le fournisseur
Pour effectuer un type de recherche du fournisseur de contacts, votre application doit disposer de l'autorisation READ_CONTACTS
.
Pour demander cela, ajoutez cet élément <uses-permission>
à votre fichier manifeste en tant qu'élément enfant de <manifest>
:
<uses-permission android:name="android.permission.READ_CONTACTS" />
Associer un contact par nom et afficher les résultats
Cette technique tente de faire correspondre une chaîne de recherche au nom d'un ou de plusieurs contacts figurant dans la table ContactsContract.Contacts
du fournisseur de contacts. Vous souhaitez généralement afficher les résultats dans un ListView
, afin de permettre à l'utilisateur de choisir parmi les contacts correspondants.
Définir des mises en page de ListView et d'éléments
Pour afficher les résultats de recherche dans un ListView
, vous avez besoin d'un fichier de mise en page principal qui définit l'ensemble de l'UI, y compris le ListView
, et d'un fichier de mise en page d'élément qui définit une ligne du ListView
. Par exemple, vous pouvez créer le fichier de mise en page principal res/layout/contacts_list_view.xml
avec le code XML suivant:
<?xml version="1.0" encoding="utf-8"?> <ListView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/list" android:layout_width="match_parent" android:layout_height="match_parent"/>
Ce fichier XML utilise le widget Android ListView
intégré android:id/list
.
Définissez le fichier de mise en page de l'élément contacts_list_item.xml
avec le code XML suivant:
<?xml version="1.0" encoding="utf-8"?> <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/text1" android:layout_width="match_parent" android:layout_height="wrap_content" android:clickable="true"/>
Ce fichier XML utilise le widget Android TextView
intégré android:text1
.
Remarque:Cette leçon ne décrit pas l'UI permettant d'obtenir une chaîne de recherche de l'utilisateur, car vous pouvez obtenir la chaîne indirectement. Par exemple, vous pouvez donner à l'utilisateur la possibilité de rechercher des contacts dont le nom correspond à une chaîne dans un message texte entrant.
Les deux fichiers de mise en page que vous avez écrits définissent une interface utilisateur qui affiche un ListView
. L'étape suivante consiste à écrire du code qui utilise cette UI pour afficher une liste de contacts.
Définir un fragment qui affiche la liste de contacts
Pour afficher la liste des contacts, commencez par définir un Fragment
chargé par un Activity
. L'utilisation d'un Fragment
est une technique plus flexible, car vous pouvez utiliser un Fragment
pour afficher la liste et un deuxième Fragment
pour afficher les détails d'un contact que l'utilisateur choisit dans la liste. Avec cette approche, vous pouvez combiner l'une des techniques présentées dans cette leçon avec celles de la leçon
Récupérer les informations d'un contact.
Pour découvrir comment utiliser un ou plusieurs objets Fragment
à partir d'un Activity
, consultez la formation
Créer une UI dynamique avec des fragments.
Pour vous aider à écrire des requêtes auprès du fournisseur de contacts, le framework Android fournit une classe de contrats appelée ContactsContract
, qui définit des constantes et des méthodes utiles pour accéder au fournisseur. Lorsque vous utilisez cette classe, vous n'avez pas besoin de définir vos propres constantes pour les URI de contenu, les noms de table ou les colonnes. Pour utiliser cette classe, incluez l'instruction suivante:
Kotlin
import android.provider.ContactsContract
Java
import android.provider.ContactsContract;
Étant donné que le code utilise un CursorLoader
pour récupérer des données auprès du fournisseur, vous devez spécifier qu'il implémente l'interface de chargeur LoaderManager.LoaderCallbacks
. Pour aider à détecter le contact que l'utilisateur sélectionne dans la liste des résultats de recherche, implémentez l'interface d'adaptateur AdapterView.OnItemClickListener
. Exemple :
Kotlin
... import android.support.v4.app.Fragment import android.support.v4.app.LoaderManager import android.widget.AdapterView ... class ContactsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor>, AdapterView.OnItemClickListener {
Java
... import android.support.v4.app.Fragment; import android.support.v4.app.LoaderManager.LoaderCallbacks; import android.widget.AdapterView; ... public class ContactsFragment extends Fragment implements LoaderManager.LoaderCallbacks<Cursor>, AdapterView.OnItemClickListener {
Définir des variables globales
Définissez les variables globales qui seront utilisées dans d'autres parties du code:
Kotlin
... /* * Defines an array that contains column names to move from * the Cursor to the ListView. */ @SuppressLint("InlinedApi") private val FROM_COLUMNS: Array<String> = arrayOf( if ((Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB)) { ContactsContract.Contacts.DISPLAY_NAME_PRIMARY } else { ContactsContract.Contacts.DISPLAY_NAME } ) /* * Defines an array that contains resource ids for the layout views * that get the Cursor column contents. The id is pre-defined in * the Android framework, so it is prefaced with "android.R.id" */ private val TO_IDS: IntArray = intArrayOf(android.R.id.text1) ... class ContactsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor>, AdapterView.OnItemClickListener { ... // Define global mutable variables // Define a ListView object lateinit var contactsList: ListView // Define variables for the contact the user selects // The contact's _ID value var contactId: Long = 0 // The contact's LOOKUP_KEY var contactKey: String? = null // A content URI for the selected contact var contactUri: Uri? = null // An adapter that binds the result Cursor to the ListView private val cursorAdapter: SimpleCursorAdapter? = null
Java
... /* * Defines an array that contains column names to move from * the Cursor to the ListView. */ @SuppressLint("InlinedApi") private final static String[] FROM_COLUMNS = { Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? ContactsContract.Contacts.DISPLAY_NAME_PRIMARY : ContactsContract.Contacts.DISPLAY_NAME }; /* * Defines an array that contains resource ids for the layout views * that get the Cursor column contents. The id is pre-defined in * the Android framework, so it is prefaced with "android.R.id" */ private final static int[] TO_IDS = { android.R.id.text1 }; // Define global mutable variables // Define a ListView object ListView contactsList; // Define variables for the contact the user selects // The contact's _ID value long contactId; // The contact's LOOKUP_KEY String contactKey; // A content URI for the selected contact Uri contactUri; // An adapter that binds the result Cursor to the ListView private SimpleCursorAdapter cursorAdapter; ...
Remarque:Étant donné que Contacts.DISPLAY_NAME_PRIMARY
nécessite Android 3.0 (version d'API 11) ou version ultérieure, définir la valeur minSdkVersion
de votre application sur 10 ou moins génère une alerte Android Lint dans Android Studio. Pour désactiver cet avertissement, ajoutez l'annotation @SuppressLint("InlinedApi")
avant la définition de FROM_COLUMNS
.
Initialiser le fragment
Initialisez le Fragment
. Ajoutez le constructeur public vide requis par le système Android et gonflez l'UI de l'objet Fragment
dans la méthode de rappel onCreateView()
.
Exemple :
Kotlin
// A UI Fragment must inflate its View override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { // Inflate the fragment layout return inflater.inflate(R.layout.contact_list_fragment, container, false) }
Java
// Empty public constructor, required by the system public ContactsFragment() {} // A UI Fragment must inflate its View @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // Inflate the fragment layout return inflater.inflate(R.layout.contact_list_fragment, container, false); }
Configurer le CursorAdapter pour la ListView
Configurez le SimpleCursorAdapter
qui lie les résultats de la recherche au ListView
. Pour obtenir l'objet ListView
qui affiche les contacts, vous devez appeler Activity.findViewById()
à l'aide de l'activité parente de Fragment
. Utilisez le Context
de l'activité parent lorsque vous appelez setAdapter()
.
Exemple :
Kotlin
override fun onActivityCreated(savedInstanceState: Bundle?) { super.onActivityCreated(savedInstanceState) ... // Gets the ListView from the View list of the parent activity activity?.also { contactsList = it.findViewById<ListView>(R.id.contact_list_view) // Gets a CursorAdapter cursorAdapter = SimpleCursorAdapter( it, R.layout.contact_list_item, null, FROM_COLUMNS, TO_IDS, 0 ) // Sets the adapter for the ListView contactsList.adapter = cursorAdapter } }
Java
public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); ... // Gets the ListView from the View list of the parent activity contactsList = (ListView) getActivity().findViewById(R.layout.contact_list_view); // Gets a CursorAdapter cursorAdapter = new SimpleCursorAdapter( getActivity(), R.layout.contact_list_item, null, FROM_COLUMNS, TO_IDS, 0); // Sets the adapter for the ListView contactsList.setAdapter(cursorAdapter); }
Définir l'écouteur de contact sélectionné
Lorsque vous affichez les résultats d'une recherche, vous souhaitez généralement autoriser l'utilisateur à sélectionner un seul contact pour un traitement ultérieur. Par exemple, lorsque l'utilisateur clique sur un contact, vous pouvez afficher son adresse sur une carte. Pour fournir cette fonctionnalité, vous avez d'abord défini l'Fragment
actuel comme écouteur de clic en spécifiant que la classe implémente AdapterView.OnItemClickListener
, comme indiqué dans la section Définir un fragment qui affiche la liste des contacts.
Pour continuer à configurer l'écouteur, liez-le à ListView
en appelant la méthode setOnItemClickListener()
dans onActivityCreated()
. Exemple :
Kotlin
fun onActivityCreated(savedInstanceState:Bundle) { ... // Set the item click listener to be the current fragment. contactsList.onItemClickListener = this ... }
Java
public void onActivityCreated(Bundle savedInstanceState) { ... // Set the item click listener to be the current fragment. contactsList.setOnItemClickListener(this); ... }
Comme vous avez spécifié que le Fragment
actuel est le OnItemClickListener
pour ListView
, vous devez maintenant implémenter la méthode requise onItemClick()
, qui gère l'événement de clic. Ce point est décrit dans une section suivante.
Définir une projection
Définissez une constante contenant les colonnes que vous souhaitez renvoyer à partir de votre requête. Chaque élément de ListView
affiche le nom à afficher du contact, qui contient la forme principale du nom du contact. Dans Android 3.0 (version 11 de l'API) et versions ultérieures, le nom de cette colonne est Contacts.DISPLAY_NAME_PRIMARY
. Dans les versions antérieures, il est Contacts.DISPLAY_NAME
.
La colonne Contacts._ID
est utilisée par le processus de liaison SimpleCursorAdapter
.
Contacts._ID
et LOOKUP_KEY
sont utilisés ensemble pour créer un URI de contenu pour le contact sélectionné par l'utilisateur.
Kotlin
... @SuppressLint("InlinedApi") private val PROJECTION: Array<out String> = arrayOf( ContactsContract.Contacts._ID, ContactsContract.Contacts.LOOKUP_KEY, if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) ContactsContract.Contacts.DISPLAY_NAME_PRIMARY else ContactsContract.Contacts.DISPLAY_NAME )
Java
... @SuppressLint("InlinedApi") private static final String[] PROJECTION = { Contacts._ID, Contacts.LOOKUP_KEY, Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? ContactsContract.Contacts.DISPLAY_NAME_PRIMARY : ContactsContract.Contacts.DISPLAY_NAME };
Définir des constantes pour les index de colonne du curseur
Pour obtenir les données d'une colonne spécifique dans une Cursor
, vous avez besoin de l'index de la colonne dans la Cursor
. Vous pouvez définir des constantes pour les indices des colonnes Cursor
, car les indices sont identiques à l'ordre des noms de colonnes dans votre projection. Exemple :
Kotlin
// The column index for the _ID column private const val CONTACT_ID_INDEX: Int = 0 // The column index for the CONTACT_KEY column private const val CONTACT_KEY_INDEX: Int = 1
Java
// The column index for the _ID column private static final int CONTACT_ID_INDEX = 0; // The column index for the CONTACT_KEY column private static final int CONTACT_KEY_INDEX = 1;
Spécifier les critères de sélection
Pour spécifier les données souhaitées, créez une combinaison d'expressions textuelles et de variables qui indiquent au fournisseur les colonnes de données à rechercher et les valeurs à trouver.
Pour l'expression textuelle, définissez une constante qui liste les colonnes de recherche. Bien que cette expression puisse également contenir des valeurs, il est préférable de les représenter à l'aide d'un espace réservé "?". Lors de la récupération, l'espace réservé est remplacé par les valeurs d'un tableau. L'utilisation de "?" comme espace réservé garantit que la spécification de recherche est générée par liaison plutôt que par compilation SQL. Cette pratique élimine la possibilité d'injection SQL malveillante. Exemple :
Kotlin
// Defines the text expression @SuppressLint("InlinedApi") private val SELECTION: String = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) "${ContactsContract.Contacts.DISPLAY_NAME_PRIMARY} LIKE ?" else "${ContactsContract.Contacts.DISPLAY_NAME} LIKE ?" ... // Defines a variable for the search string private val searchString: String = ... // Defines the array to hold values that replace the ? private val selectionArgs = arrayOf<String>(searchString)
Java
// Defines the text expression @SuppressLint("InlinedApi") private static final String SELECTION = Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? Contacts.DISPLAY_NAME_PRIMARY + " LIKE ?" : Contacts.DISPLAY_NAME + " LIKE ?"; // Defines a variable for the search string private String searchString; // Defines the array to hold values that replace the ? private String[] selectionArgs = { searchString };
Définir la méthode onItemClick()
Dans une section précédente, vous avez défini l'écouteur de clics de l'élément pour ListView
.
Implémentez maintenant l'action pour l'écouteur en définissant la méthode AdapterView.OnItemClickListener.onItemClick()
:
Kotlin
override fun onItemClick(parent: AdapterView<*>, view: View?, position: Int, id: Long) { // Get the Cursor val cursor: Cursor? = (parent.adapter as? CursorAdapter)?.cursor?.apply { // Move to the selected contact moveToPosition(position) // Get the _ID value contactId = getLong(CONTACT_ID_INDEX) // Get the selected LOOKUP KEY contactKey = getString(CONTACT_KEY_INDEX) // Create the contact's content Uri contactUri = ContactsContract.Contacts.getLookupUri(contactId, mContactKey) /* * You can use contactUri as the content URI for retrieving * the details for a contact. */ } }
Java
@Override public void onItemClick( AdapterView<?> parent, View item, int position, long rowID) { // Get the Cursor Cursor cursor = parent.getAdapter().getCursor(); // Move to the selected contact cursor.moveToPosition(position); // Get the _ID value contactId = cursor.getLong(CONTACT_ID_INDEX); // Get the selected LOOKUP KEY contactKey = cursor.getString(CONTACT_KEY_INDEX); // Create the contact's content Uri contactUri = Contacts.getLookupUri(contactId, mContactKey); /* * You can use contactUri as the content URI for retrieving * the details for a contact. */ }
Initialiser le chargeur
Étant donné que vous utilisez un CursorLoader
pour récupérer des données, vous devez initialiser le thread en arrière-plan et d'autres variables qui contrôlent la récupération asynchrone. Effectuez l'initialisation dans onCreate()
comme indiqué dans l'exemple suivant:
Kotlin
class ContactsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor> { ... override fun onCreate(savedInstanceState: Bundle?) { // Always call the super method first super.onCreate(savedInstanceState) ... // Initializes the loader loaderManager.initLoader(0, null, this)
Java
public class ContactsFragment extends Fragment implements LoaderManager.LoaderCallbacks<Cursor> { ... // Called just before the Fragment displays its UI @Override public void onCreate(Bundle savedInstanceState) { // Always call the super method first super.onCreate(savedInstanceState); ... // Initializes the loader getLoaderManager().initLoader(0, null, this);
Implémenter onCreateLoader()
Implémentez la méthode onCreateLoader()
, qui est appelée par le framework du chargeur immédiatement après l'appel de initLoader()
.
Dans onCreateLoader()
, configurez le modèle de chaîne de recherche. Pour transformer une chaîne en modèle, insérez des caractères "%" (pourcentage) pour représenter une séquence de zéro ou plusieurs caractères, ou des caractères "_" (underscore) pour représenter un seul caractère, ou les deux. Par exemple, le modèle "%Jefferson%" correspond à la fois à "Thomas Jefferson" et à "Jefferson Davis".
Renvoyez une nouvelle CursorLoader
à partir de la méthode. Pour l'URI du contenu, utilisez Contacts.CONTENT_URI
.
Cet URI fait référence à l'ensemble de la table, comme illustré dans l'exemple suivant:
Kotlin
... override fun onCreateLoader(loaderId: Int, args: Bundle?): Loader<Cursor> { /* * Makes search string into pattern and * stores it in the selection array */ selectionArgs[0] = "%$mSearchString%" // Starts the query return activity?.let { return CursorLoader( it, ContactsContract.Contacts.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ) } ?: throw IllegalStateException() }
Java
... @Override public Loader<Cursor> onCreateLoader(int loaderId, Bundle args) { /* * Makes search string into pattern and * stores it in the selection array */ selectionArgs[0] = "%" + searchString + "%"; // Starts the query return new CursorLoader( getActivity(), ContactsContract.Contacts.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ); }
Implémenter onLoadFinished() et onLoaderReset()
Implémentez la méthode onLoadFinished()
. Le framework de chargement appelle onLoadFinished()
lorsque le fournisseur de contacts renvoie les résultats de la requête. Dans cette méthode, placez le résultat Cursor
dans SimpleCursorAdapter
. Les résultats de recherche sont automatiquement mis à jour dans ListView
:
Kotlin
override fun onLoadFinished(loader: Loader<Cursor>, cursor: Cursor) { // Put the result Cursor in the adapter for the ListView cursorAdapter?.swapCursor(cursor) }
Java
@Override public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) { // Put the result Cursor in the adapter for the ListView cursorAdapter.swapCursor(cursor); }
La méthode onLoaderReset()
est appelée lorsque le framework de chargement détecte que le résultat Cursor
contient des données obsolètes. Supprimez la référence SimpleCursorAdapter
à l'Cursor
existante. Sinon, le framework de chargeur ne recyclera pas Cursor
, ce qui entraînera une fuite de mémoire. Exemple :
Kotlin
override fun onLoaderReset(loader: Loader<Cursor>) { // Delete the reference to the existing Cursor cursorAdapter?.swapCursor(null) }
Java
@Override public void onLoaderReset(Loader<Cursor> loader) { // Delete the reference to the existing Cursor cursorAdapter.swapCursor(null); }
Vous disposez désormais des éléments clés d'une application qui met en correspondance une chaîne de recherche avec des noms de contact et renvoie le résultat dans un ListView
. L'utilisateur peut cliquer sur le nom d'un contact pour le sélectionner.
Cela déclenche un écouteur, dans lequel vous pouvez continuer à travailler avec les données du contact. Par exemple, vous pouvez récupérer les coordonnées du contact. Pour savoir comment procéder, passez à la leçon suivante, Récupérer les informations d'un contact.
Pour en savoir plus sur les interfaces utilisateur de recherche, consultez le guide de l'API Créer une interface de recherche.
Les sections restantes de cette leçon présentent d'autres façons de trouver des contacts dans le fournisseur de contacts.
Faire correspondre un contact avec un type de données spécifique
Cette technique vous permet de spécifier le type de données que vous souhaitez faire correspondre. La récupération par nom est un exemple spécifique de ce type de requête, mais vous pouvez également le faire pour tous les types de données détaillées associées à un contact. Par exemple, vous pouvez récupérer les contacts ayant un code postal spécifique. Dans ce cas, la chaîne de recherche doit correspondre aux données stockées dans une ligne de code postal.
Pour implémenter ce type de récupération, implémentez d'abord le code suivant, comme indiqué dans les sections précédentes:
- Demandez l'autorisation de lire le fournisseur.
- Définissez ListView et les mises en page des éléments.
- Définissez un fragment qui affiche la liste des contacts.
- Définissez des variables globales.
- Initialisez le fragment.
- Configurez le CursorAdapter pour la ListView.
- Définissez l'écouteur de contact sélectionné.
-
Définissez des constantes pour les index de colonne de curseur.
Bien que vous récupériez les données d'une table différente, l'ordre des colonnes dans la projection est le même. Vous pouvez donc utiliser les mêmes index pour le curseur.
- Définissez la méthode onItemClick().
- Initialisez le chargeur.
- Implémentez onLoadFinished() et onLoaderReset().
Les étapes suivantes vous montrent le code supplémentaire dont vous avez besoin pour faire correspondre une chaîne de recherche à un type particulier de données détaillées et afficher les résultats.
Choisir le type de données et la table
Pour rechercher un type de données détaillées particulier, vous devez connaître la valeur du type MIME personnalisé pour le type de données. Chaque type de données a une valeur de type MIME unique définie par une constante CONTENT_ITEM_TYPE
dans la sous-classe de ContactsContract.CommonDataKinds
associée au type de données.
Les sous-classes ont des noms qui indiquent leur type de données. Par exemple, la sous-classe des données de messagerie est ContactsContract.CommonDataKinds.Email
, et le type MIME personnalisé pour les données de messagerie est défini par la constante Email.CONTENT_ITEM_TYPE
.
Utilisez le tableau ContactsContract.Data
pour votre recherche. Toutes les constantes dont vous avez besoin pour votre projection, votre clause de sélection et votre ordre de tri sont définies dans cette table ou héritées par celle-ci.
Définir une projection
Pour définir une projection, choisissez une ou plusieurs des colonnes définies dans ContactsContract.Data
, ou les classes dont elle hérite. Le fournisseur de contacts effectue une jointure implicite entre ContactsContract.Data
et d'autres tables avant de renvoyer des lignes. Exemple :
Kotlin
@SuppressLint("InlinedApi") private val PROJECTION: Array<out String> = arrayOf( /* * The detail data row ID. To make a ListView work, * this column is required. */ ContactsContract.Data._ID, // The primary display name if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) ContactsContract.Data.DISPLAY_NAME_PRIMARY else ContactsContract.Data.DISPLAY_NAME, // The contact's _ID, to construct a content URI ContactsContract.Data.CONTACT_ID, // The contact's LOOKUP_KEY, to construct a content URI ContactsContract.Data.LOOKUP_KEY )
Java
@SuppressLint("InlinedApi") private static final String[] PROJECTION = { /* * The detail data row ID. To make a ListView work, * this column is required. */ ContactsContract.Data._ID, // The primary display name Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB ? ContactsContract.Data.DISPLAY_NAME_PRIMARY : ContactsContract.Data.DISPLAY_NAME, // The contact's _ID, to construct a content URI ContactsContract.Data.CONTACT_ID, // The contact's LOOKUP_KEY, to construct a content URI ContactsContract.Data.LOOKUP_KEY // A permanent link to the contact };
Définir des critères de recherche
Pour rechercher une chaîne dans un type de données particulier, créez une clause de sélection à partir des éléments suivants:
-
Nom de la colonne contenant votre chaîne de recherche. Ce nom varie selon le type de données. Vous devez donc trouver la sous-classe de
ContactsContract.CommonDataKinds
qui correspond au type de données, puis choisir le nom de la colonne dans cette sous-classe. Par exemple, pour rechercher des adresses e-mail, utilisez la colonneEmail.ADDRESS
. - La chaîne de recherche proprement dite, représentée par le caractère "?" dans la clause de sélection.
-
Nom de la colonne contenant la valeur du type MIME personnalisé. Ce nom est toujours
Data.MIMETYPE
. -
Valeur du type MIME personnalisé pour le type de données. Comme décrit précédemment, il s'agit de la constante
CONTENT_ITEM_TYPE
dans la sous-classeContactsContract.CommonDataKinds
. Par exemple, la valeur de type MIME pour les données de messagerie estEmail.CONTENT_ITEM_TYPE
. Placez la valeur entre guillemets simples en concaténant un caractère "'
" (guillemet simple) au début et à la fin de la constante. Sinon, le fournisseur interprète la valeur comme un nom de variable plutôt que comme une valeur de chaîne. Vous n'avez pas besoin d'utiliser un espace réservé pour cette valeur, car vous utilisez une constante plutôt qu'une valeur fournie par l'utilisateur.
Exemple :
Kotlin
/* * Constructs search criteria from the search string * and email MIME type */ private val SELECTION: String = /* * Searches for an email address * that matches the search string */ "${Email.ADDRESS} LIKE ? AND " + /* * Searches for a MIME type that matches * the value of the constant * Email.CONTENT_ITEM_TYPE. Note the * single quotes surrounding Email.CONTENT_ITEM_TYPE. */ "${ContactsContract.Data.MIMETYPE } = '${Email.CONTENT_ITEM_TYPE}'"
Java
/* * Constructs search criteria from the search string * and email MIME type */ private static final String SELECTION = /* * Searches for an email address * that matches the search string */ Email.ADDRESS + " LIKE ? " + "AND " + /* * Searches for a MIME type that matches * the value of the constant * Email.CONTENT_ITEM_TYPE. Note the * single quotes surrounding Email.CONTENT_ITEM_TYPE. */ ContactsContract.Data.MIMETYPE + " = '" + Email.CONTENT_ITEM_TYPE + "'";
Ensuite, définissez les variables devant contenir l'argument de sélection:
Kotlin
private var searchString: String? = null private val selectionArgs: Array<String> = arrayOf("")
Java
String searchString; String[] selectionArgs = { "" };
Implémenter onCreateLoader()
Maintenant que vous avez spécifié les données que vous souhaitez et comment les trouver, définissez une requête dans votre implémentation de onCreateLoader()
.
Renvoyez un nouvel élément CursorLoader
à partir de cette méthode, en utilisant votre projection, votre expression de texte de sélection et votre tableau de sélection comme arguments. Pour un URI de contenu, utilisez Data.CONTENT_URI
. Exemple :
Kotlin
override fun onCreateLoader(id: Int, args: Bundle?): Loader<Cursor> { // OPTIONAL: Makes search string into pattern searchString = "%$mSearchString%" searchString?.also { // Puts the search string into the selection criteria selectionArgs[0] = it } // Starts the query return activity?.let { CursorLoader( it, ContactsContract.Data.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ) } ?: throw IllegalStateException() }
Java
@Override public Loader<Cursor> onCreateLoader(int loaderId, Bundle args) { // OPTIONAL: Makes search string into pattern searchString = "%" + searchString + "%"; // Puts the search string into the selection criteria selectionArgs[0] = searchString; // Starts the query return new CursorLoader( getActivity(), Data.CONTENT_URI, PROJECTION, SELECTION, selectionArgs, null ); }
Ces extraits de code constituent la base d'une recherche inverse simple basée sur un type spécifique de données détaillées. Il s'agit de la meilleure technique à utiliser si votre application se concentre sur un type de données particulier, comme les e-mails, et que vous souhaitez permettre aux utilisateurs d'obtenir les noms associés à une donnée.
Faire correspondre un contact selon n'importe quel type de données
La récupération d'un contact en fonction de n'importe quel type de données renvoie des contacts si l'une de leurs données correspond à la chaîne de recherche, y compris le nom, l'adresse e-mail, l'adresse postale, le numéro de téléphone, etc. Vous obtenez ainsi un ensemble étendu de résultats de recherche. Par exemple, si la chaîne de recherche est "Doe", la recherche de n'importe quel type de données renvoie le contact "Jean Dupont", mais également les contacts qui vivent sur "Doe Street".
Pour implémenter ce type de récupération, implémentez d'abord le code suivant, comme indiqué dans les sections précédentes:
- Demandez l'autorisation de lire le fournisseur.
- Définissez ListView et les mises en page des éléments.
- Définissez un fragment qui affiche la liste des contacts.
- Définissez des variables globales.
- Initialisez le fragment.
- Configurez le CursorAdapter pour la ListView.
- Définissez l'écouteur de contacts sélectionné.
- Définissez une projection.
-
Définissez des constantes pour les index de colonne du curseur.
Pour ce type de récupération, vous utilisez la même table que dans la section Associer un contact par nom et lister les résultats. Utilisez également les mêmes index de colonne.
- Définissez la méthode onItemClick().
- Initialisez le chargeur.
- Implémentez onLoadFinished() et onLoaderReset().
Les étapes suivantes vous montrent le code supplémentaire dont vous avez besoin pour faire correspondre une chaîne de recherche à n'importe quel type de données et afficher les résultats.
Supprimer les critères de sélection
Ne définissez pas les constantes SELECTION
ni la variable mSelectionArgs
.
Ils ne sont pas utilisés dans ce type de récupération.
Implémenter onCreateLoader()
Implémentez la méthode onCreateLoader()
, qui renvoie un nouveau CursorLoader
.
Vous n'avez pas besoin de convertir la chaîne de recherche en modèle, car le fournisseur de contacts le fait automatiquement. Utilisez Contacts.CONTENT_FILTER_URI
comme URI de base, puis ajoutez votre chaîne de recherche en appelant Uri.withAppendedPath()
. L'utilisation de cet URI déclenche automatiquement la recherche de n'importe quel type de données, comme illustré dans l'exemple suivant:
Kotlin
override fun onCreateLoader(loaderId: Int, args: Bundle?): Loader<Cursor> { /* * Appends the search string to the base URI. Always * encode search strings to ensure they're in proper * format. */ val contentUri: Uri = Uri.withAppendedPath( ContactsContract.Contacts.CONTENT_FILTER_URI, Uri.encode(searchString) ) // Starts the query return activity?.let { CursorLoader( it, contentUri, PROJECTION2, null, null, null ) } ?: throw IllegalStateException() }
Java
@Override public Loader<Cursor> onCreateLoader(int loaderId, Bundle args) { /* * Appends the search string to the base URI. Always * encode search strings to ensure they're in proper * format. */ Uri contentUri = Uri.withAppendedPath( Contacts.CONTENT_FILTER_URI, Uri.encode(searchString)); // Starts the query return new CursorLoader( getActivity(), contentUri, PROJECTION, null, null, null ); }
Ces extraits de code constituent la base d'une application qui effectue une recherche étendue dans le fournisseur de contacts. Cette technique est utile pour les applications qui souhaitent implémenter une fonctionnalité semblable à l'écran de liste de contacts de l'application People.