web-dev-qa-db-ja.com

Intent.ACTION_PICKから返されたURIから特定の連絡先情報を取得します

人(具体的には、子供の親または保護者)を表すデータ型を持つAndroidアプリを作成しています)。関連するデータを「インポート」できるようにしたいと思います。 Androidデバイスの連絡先データベースのフィールド。(これはオプションである必要があります。つまり、親/保護者がすでに連絡先データベースにある必要はなく、連絡先も必要ありません。新しい親/保護者が追加された場合、データベースは更新されます。)

これまで、特定の連絡先を選択するための新しいインテントを開始するコードを記述しました(Intent.ACTION_PICKを使用)。次に、データベース内の特定の連絡先を表すURIを取得します。

残念ながら、次のステップが何であるかわかりません。これは世界で最も簡単なことのようですが、そうではないようです。 Android開発者向けWebサイトのドキュメントを読み、複数のAndroid本を読みました。喜びはありません。

私が入手したい具体的な情報は次のとおりです。

  1. 連絡先の名前(可能であれば最初と最後を別々に)

  2. 連絡先の(プライマリ)メールアドレス

  3. 連絡先の携帯電話番号

これは、ContentResolverを使用してクエリを実行することで可能になるはずですが、インテントから返されたURIを使用してこれを行う方法がわかりません。ほとんどのドキュメントは、連絡先のURIではなく、連絡先IDを持っていることを前提としています。また、これが私が望むことを行う正しい方法でさえあると仮定すると、クエリのプロジェクションにどのようなフィールドを入れることができるのかわかりません。

これが私の開始コードです:

// In a button's onClick event handler:
Intent intent = new Intent(Intent.ACTION_PICK, ContactsContract.Contacts.CONTENT_URI);
startActivityForResult(intent, PICK_CONTACT);

// In onActivityResult:
if (resultCode == RESULT_OK) {
    if (requestCode == PICK_CONTACT) {
        contactURI = data.getData();
        // NOW WHAT?
    }
}
15
Karl Giesing

結局のところ、これを行うにはより良い方法があります。

前述したように、ContactsContract.Contacts.EntityクラスはAPI 11まで使用できませんでした。ただし、 ContactsContract.Data クラスはAPI 5で使用可能であり、ほぼ同じクラスで使用できます。 Entityクラスを使用する方法。

コードを更新しました。これは、Entityクラスのコードと非常によく似ており、ほぼ同じように機能します。ただし、Gingerbreadを実行している電話でテストしたところ、問題なく動作します。

私がしなければならなかった1つの変更はこれです:最初のクエリからContactsContract.Data.RAW_CONTACT_IDを取得する方法がないようで、そのIDは同じではありませんあなたが例えばから得るIDとしてContactsContract.Contacts._ID。代わりに、ContactsContract.Contacts.DISPLAY_NAMEconstantについて問い合わせました。これは、ほぼすべてのContactsContractクラスで一貫しています。

動作するコードは次のとおりです。

        Cursor cursor;  // Cursor object
        String mime;    // MIME type
        int dataIdx;    // Index of DATA1 column
        int mimeIdx;    // Index of MIMETYPE column
        int nameIdx;    // Index of DISPLAY_NAME column

        // Get the name
        cursor = getContentResolver().query(params[0],
                new String[] { ContactsContract.Contacts.DISPLAY_NAME },
                null, null, null);
        if (cursor.moveToFirst()) {
            nameIdx = cursor.getColumnIndex(
                    ContactsContract.Contacts.DISPLAY_NAME);
            name = cursor.getString(nameIdx);

            // Set up the projection
            String[] projection = {
                    ContactsContract.Data.DISPLAY_NAME,
                    ContactsContract.Contacts.Data.DATA1,
                    ContactsContract.Contacts.Data.MIMETYPE };

            // Query ContactsContract.Data
            cursor = getContentResolver().query(
                    ContactsContract.Data.CONTENT_URI, projection,
                    ContactsContract.Data.DISPLAY_NAME + " = ?",
                    new String[] { name },
                    null);

            if (cursor.moveToFirst()) {
                // Get the indexes of the MIME type and data
                mimeIdx = cursor.getColumnIndex(
                        ContactsContract.Contacts.Data.MIMETYPE);
                dataIdx = cursor.getColumnIndex(
                        ContactsContract.Contacts.Data.DATA1);

                // Match the data to the MIME type, store in variables
                do {
                    mime = cursor.getString(mimeIdx);
                    if (ContactsContract.CommonDataKinds.Email
                            .CONTENT_ITEM_TYPE.equalsIgnoreCase(mime)) {
                        email = cursor.getString(dataIdx);
                    }
                    if (ContactsContract.CommonDataKinds.Phone
                            .CONTENT_ITEM_TYPE.equalsIgnoreCase(mime)) {
                        phone = cursor.getString(dataIdx);
                        phone = PhoneNumberUtils.formatNumber(phone);
                    }
                } while (cursor.moveToNext());
            }
        }
17
Karl Giesing

さて、たくさん掘り下げた後、私は自分が答えだと信じているものを見つけました。私が見つけた解決策は、使用しているAndroid APIレベルによって異なります。ただし、それらはまったくきれいではないので、より良い解決策があれば知りたいです。

いずれの場合も、最初のステップは、Intent.ACTION_PICKから返されたURIに対してクエリを実行して、連絡先のIDを取得することです。ここにいる間に、表示名と、連絡先に電話番号があるかどうかを表す文字列も取得する必要があります。 (最新のソリューションでは必要ありませんが、レガシーソリューションでは必要になります。)

String id, name, phone, hasPhone;
int idx;
Cursor cursor = getContentResolver().query(contactUri, null, null, null, null);
if (cursor.moveToFirst()) {
    idx = cursor.getColumnIndex(ContactsContract.Contacts._ID);
    id = cursor.getString(idx);

    idx = cursor.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME);
    name = cursor.getString(idx);

    idx = cursor.getColumnIndex(ContactsContract.Contacts.HAS_PHONE_NUMBER);
    hasPhone = cursor.getString(idx);
}

レコードの場合、このURIから返される列は、ほとんどのContactsContract.Profile クラスの定数で表されるフィールドです(他のインターフェースから継承された定数を含む)。 PHOTO_FILE_ID、PHOTO_THUMBNAIL_URI、またはPHOTO_URIは含まれていません(ただし、PHOTO_IDが含まれています)。

IDを取得したので、関連するデータを取得する必要があります。最初の(そして最も単純な)解決策は、エンティティを照会することです。エンティティクエリは、連絡先または未加工の連絡先のすべての連絡先データを一度に取得します。各行は、 ContactsContract.Contacts.Entity の定数を使用してアクセスされる単一のRawContactを表します。通常は、RAW_CONTACT_ID、DATA1、およびMIMETYPEのみに関心があります。ただし、姓名を別々に使用する場合は、Name MIMEタイプは、DATA2に名を保持し、DATA3に姓を保持します。

MIMETYPE列を ContactsContract.CommonDataKinds 定数と照合することにより、変数をロードします。たとえば、電子メールのMIMEタイプは ContactsContract.CommonDataKinds.Email.CONTENT_ITEM_TYPE にあります。

// Build the Entity URI.
Uri.Builder b = Uri.withAppendedPath(ContactsContract.Contacts.CONTENT_URI, id).buildUpon();
b.appendPath(ContactsContract.Contacts.Entity.CONTENT_DIRECTORY);
URI contactUri = b.build();

// Create the projection (SQL fields) and sort order.
String[] projection = {
        ContactsContract.Contacts.Entity.RAW_CONTACT_ID,
        ContactsContract.Contacts.Entity.DATA1,
        ContactsContract.Contacts.Entity.MIMETYPE };
String sortOrder = ContactsContract.Contacts.Entity.RAW_CONTACT_ID + " ASC";
cursor = getContentResolver().query(contactUri, projection, null, null, sortOrder);

String mime;
int mimeIdx = cursor.getColumnIndex(ContactsContract.Contacts.Entity.MIMETYPE);
int dataIdx = cursor.getColumnIndex(ContactsContract.Contacts.Entity.DATA1);
if (cursor.moveToFirst()) {
    do {
        mime = cursor.getString(mimeIdx);
        if (mime.equalsIgnoreCase(ContactsContract.CommonDataKinds.Email.CONTENT_ITEM_TYPE)) {
            email = cursor.getString(dataIdx);
        }
        if (mime.equalsIgnoreCase(ContactsContract.CommonDataKinds.Phone.CONTENT_ITEM_TYPE)) {
            phone = cursor.getString(dataIdx);
        }
        // ...etc.
    } while (cursor.moveToNext());
}

残念ながら、エンティティはAPI 11(Android 3.0、Honeycomb)まで導入されませんでした。つまり、このコードは、市場にあるAndroidデバイスの約65%と互換性がありません(この記事の執筆時点)。試してみると、URIから IllegalArgumentException が取得されます。

2番目の解決策は、クエリ文字列を作成し、使用するデータ型ごとに1つのクエリを作成することです。

// Get phone number - if they have one
if ("1".equalsIgnoreCase(hasPhone)) {
    cursor = getContentResolver().query(
            ContactsContract.CommonDataKinds.Phone.CONTENT_URI,
            null,
            ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = "+ id, 
            null, null);
    if (cursor.moveToFirst()) {
        colIdx = cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER);
        phone = cursor.getString(colIdx);
    }
    cursor.close();
}

// Get email address
cursor = getContentResolver().query(
        ContactsContract.CommonDataKinds.Email.CONTENT_URI,
        null,
        ContactsContract.CommonDataKinds.Email.CONTACT_ID + " = " + id,
        null, null);
if (cursor.moveToFirst()) {
    colIdx = cursor.getColumnIndex(ContactsContract.CommonDataKinds.Email.ADDRESS);
    email = cursor.getString(colIdx);
}
cursor.close();

// ...etc.

明らかに、この方法では多くの個別のデータベースクエリが発生するため、効率上の理由からお勧めしません。

私が思いついた解決策は、エンティティクエリを使用するバージョンを試し、IllegalArgumentExceptionをキャッチし、catchブロック内にレガシーコードを配置することです。

try {
    // Build the Entity URI.
    Uri.Builder b = Uri.withAppendedPath(ContactsContract.Contacts.CONTENT_URI, id).buildUpon();
    b.appendPath(ContactsContract.Contacts.Entity.CONTENT_DIRECTORY);
        // ...etc...
} catch (IllegalArgumentException e) {
    // Get phone number - if they have one
    if ("1".equalsIgnoreCase(hasPhone)) {   
        // ...etc...
} finally {
    // If you want to display the info in GUI objects, put the code here
}

これが誰かに役立つことを願っています。そして、繰り返しますが、これを行うためのより良い方法があれば、私はすべての耳です。

18
Karl Giesing
//Add a permission to read contacts data to your application manifest.
<uses-permission Android:name="Android.permission.READ_CONTACTS"/>

//Use Intent.ACTION_PICK in your Activity 
Intent contactPickerIntent = new Intent(Intent.ACTION_PICK,
                ContactsContract.CommonDataKinds.Phone.CONTENT_URI);
        startActivityForResult(contactPickerIntent, RESULT_PICK_CONTACT);

//Then Override the onActivityResult() and retrieve the ID,Phone number and Name in the data. 
 @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        // check whether the result is ok
        if (resultCode == RESULT_OK) {
            // Check for the request code, we might be usign multiple startActivityForReslut
            switch (requestCode) {
            case RESULT_PICK_CONTACT:
               Cursor cursor = null;
        try {
            String phoneNo = null ;
            String name = null;           
            Uri uri = data.getData();            
            cursor = getContentResolver().query(uri, null, null, null, null);
            cursor.moveToFirst();           
            int  phoneIndex =cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER);
            phoneNo = cursor.getString(phoneIndex); 
            textView2.setText(phoneNo);
        } catch (Exception e) {
            e.printStackTrace();
        }
                break;
            }
        } else {
            Log.e("MainActivity", "Failed to pick contact");
        }
    }

注:Android 2.3(APIレベル9)の前に、連絡先プロバイダー(上記のような)でクエリを実行するには、アプリでREAD_CONTACTS権限を宣言する必要があります(セキュリティと権限を参照)。 、Android 2.3で始まり、Contacts/Peopleアプリは、結果を返すときに連絡先プロバイダーから読み取るための一時的なアクセス許可をアプリに付与します。一時的なアクセス許可は、要求された特定の連絡先にのみ適用されます。したがって、READ_CONTACTS権限を宣言しない限り、インテントのUriで指定された連絡先以外の連絡先にクエリを実行することはできません。

2
Ravi