2016-12-20 1 views
1

Ситуация: Я добавил пользовательское действие для контакта в Android в соответствии с инструкциями в this question и related sample app on GitHub. При нажатии я хочу набрать этот контакт в своем приложении. Я могу успешно восстановить контакт в Activity, который открывается при нажатии пользовательского действия.Как получить номера телефонов для контакта, нажав на настраиваемое поле в приложении «Натуральные контакты» на Android?

Я исполняю это:

Cursor cursor = context.getContentResolver().query(data, null, null, null, null); 
if (cursor != null) { 
    newIntent = true; 
    contact = LocalContactAsync.getContacts(cursor, context.getContentResolver()).get(0); 
    cursor.close(); 
} 

И data я получить от Android является:

содержание: //com.android.contacts/data/2849

Примечание число 2849 в конце, это не родной идентификатор контакта. Нативный идентификатор контакта - 459. Я в состоянии успешно восстановить контакт выполняет этот запрос, следующие возвращает данные: '2849'

cur.getString(cur.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME)) ; 

'образец samplee' -returns которым

cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts._ID); 

-returns является правильным

Но хотя это верно:

cursor.getInt(cur.getColumnIndex(
      ContactsContract.Contacts.HAS_PHONE_NUMBER)) > 0) 

Следующая функция возвращает пустой курсор:

Cursor pCur = cr.query(
       ContactsContract.CommonDataKinds.Phone.CONTENT_URI, 
       null, 
       ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = ?", 
       new String[]{id}, null); 

-id = 2849 в этом случае, но если я заполняю 459 я получить нужное количество телефонных номеров

Реальный контакт имеет 3 номера, чтобы он должен вернуть 3 номера.

Как я могу это исправить?

Отредактировано:

Это, как я получить номера, чтобы быть ясно: я получаю правильное название, но следующий запрос возвращает null в то время как контакт имеет номер.

ArrayList<Number> numbers = new ArrayList<>(); 
    if (cur.getInt(cur.getColumnIndex(
      ContactsContract.Contacts.HAS_PHONE_NUMBER)) > 0) { 
     Cursor pCur = cr.query(
       ContactsContract.CommonDataKinds.Phone.CONTENT_URI, 
       null, 
       ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = ?", 
       new String[]{id}, null); 
     while (pCur.moveToNext()) { 
      numbers.add(new nl.coffeeit.clearvoxnexxt.objects.dto.Number(pCur.getString(pCur.getColumnIndex(
        ContactsContract.CommonDataKinds.Phone.NUMBER)))); 
     } 
     pCur.close(); 
    } 
    return numbers; 

Пожалуйста, обратите внимание, что я не просить намерения, я получаю его через настраиваемое действие, которое добавляется к нативной контакта, как Viber и WhatsApp сделать:

custom action android contact

Полный код LocalContacts асинхронный:

private static final String TAG = "LocalContactAsync"; 
private static List<Contact> contacts; 

private Context context; 
private boolean refreshOtherFragments; 
private boolean renew;  

private synchronized List<Contact> getContacts(Context context) { 
    if (!renew && contacts != null) { 
     return contacts; 
    } 
    ContentResolver cr = context.getContentResolver(); 
    Cursor cur = cr.query(ContactsContract.Contacts.CONTENT_URI, 
      null, null, null, null); 

    if (cur != null && cur.getCount() > 0) { 
     contacts = getContacts(cur, cr); 
     cur.close(); 
     return contacts; 
    } 
    if (cur != null) { 
     cur.close(); 
    } 
    return new ArrayList<>(); 
} 

public static List<Contact> getContacts(Cursor cur, ContentResolver cr) { 
    List<Contact> contacts = new ArrayList<>(); 
    while (cur.moveToNext()) { 
     String id = getId(cur); 
     String name = getName(cur); 
     ArrayList<Number> numbers = getNumbers(cur, cr, id); 
     if (name != null) { 
      contacts.add(new Contact(id, name, numbers)); 
     } 
    } 
    Log.d(TAG, "amount of contacts" + contacts.size()); 
    return contacts; 
} 

private static ArrayList<Number> getNumbers(Cursor cur, ContentResolver cr, String id) { 
    ArrayList<Number> numbers = new ArrayList<>(); 
    if (cur.getInt(cur.getColumnIndex(
      ContactsContract.Contacts.HAS_PHONE_NUMBER)) > 0) { 
     Cursor pCur = cr.query(
       ContactsContract.CommonDataKinds.Phone.CONTENT_URI, 
       null, 
       ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = ?", 
       new String[]{id}, null); 
     while (pCur.moveToNext()) { 
      numbers.add(getNumber(pCur)); 
     } 
     pCur.close(); 
    } 
    return numbers; 
} 

private static Number getNumber(Cursor pCur) { 
    return new nl.coffeeit.clearvoxnexxt.objects.dto.Number(pCur.getString(pCur.getColumnIndex(
      ContactsContract.CommonDataKinds.Phone.NUMBER))); 
} 

private static String getId(Cursor cur) { 
    return cur.getString(
      cur.getColumnIndex(ContactsContract.Contacts._ID)); 
} 

private static String getName(Cursor cur) { 
    return cur.getString(cur.getColumnIndex(
      ContactsContract.Contacts.DISPLAY_NAME)); 
} 

Код для DTO Номер:

public class Number implements Parcelable, Serializable { 

    @SerializedName("number") 
    @Expose 
    public String number; 
    @SerializedName("type") 
    @Expose 
    public String type = ""; 
    @SerializedName("inherited") 
    @Expose 
    public Boolean inherited = false; 

    public Number(String number) { 
     this.number = number; 
    } 

    protected Number(Parcel in) { 
     number = in.readString(); 
     type = in.readString(); 
     byte inheritedVal = in.readByte(); 
     inherited = inheritedVal == 0x02 ? null : inheritedVal != 0x00; 
    } 

    @Override 
    public int describeContents() { 
     return 0; 
    } 

    @Override 
    public void writeToParcel(Parcel dest, int flags) { 
     dest.writeString(number); 
     dest.writeString(type); 
     if (inherited == null) { 
      dest.writeByte((byte) (0x02)); 
     } else { 
      dest.writeByte((byte) (inherited ? 0x01 : 0x00)); 
     } 
    } 

    @SuppressWarnings("unused") 
    public static final Parcelable.Creator<Number> CREATOR = new Parcelable.Creator<Number>() { 
     @Override 
     public Number createFromParcel(Parcel in) { 
      return new Number(in); 
     } 

     @Override 
     public Number[] newArray(int size) { 
      return new Number[size]; 
     } 
    }; 

    public Number setNumber(String number) { 
     this.number = number; 
     return this; 
    } 
} 

ответ

1

Первое, что нужно заметить, что призыв к контактам Picker, как это:

Intent intent = new Intent(Intent.ACTION_PICK, ContactsContract.Contacts.CONTENT_URI); 

возвратит Uri, как это:

content://com.android.contacts/contacts/lookup/3163r328-4D2941473F314D2941473F3131/328 

Предпоследняя путь (3163r .. ..) является ключ поиска, а 328 - NAME_RAW_ID.

Сравните это с намерением, которое вы получаете от sample application. Это содержит Ури, который выглядит следующим образом:

content://com.android.contacts/data/2849 

Как вы сказали, вызывая распознаватель контента с этим Ури не достаточно, чтобы получить номера телефонов, хотя оно может быть использовано для извлечения имени контакта и Я бы. Поэтому мы будем использовать неполный Intent Uri, чтобы построить новый Lookup Uri, который мы можем использовать для получения номеров телефонов.

Давайте добавим следующие методы к вашему LocalContactAsync (я не все, что вы сделали до сих пор рефакторинга, я просто добавить в стиль, который вы использовали):

public static Uri getLookupUri(Cursor cur) { 
    return getContentUri(getLookupKey(cur), getNameRawId(cur)); 
} 

private static String getLookupKey(Cursor cur) { 
    return cur.getString(
      cur.getColumnIndex(ContactsContract.Contacts.LOOKUP_KEY)); 
} 

private static String getNameRawId(Cursor cur) { 
    return cur.getString(cur.getColumnIndex(ContactsContract.Contacts.NAME_RAW_CONTACT_ID)); 
} 

private static Uri getContentUri(String lookupKey, String nameRawId) { 
    return new Uri.Builder() 
      .scheme("content") 
      .authority("com.android.contacts") 
      .appendPath("contacts") 
      .appendPath("lookup") 
      .appendPath(lookupKey) 
      .appendPath(nameRawId) 
      .build(); 
} 

Давайте изменить ViewingActivity внутри пример приложения, чтобы он действительно извлекал контактные данные. Теперь мы можем сделать это с помощью следующего кода внутри onResume():

@Override 
protected void onResume() { 
    super.onResume(); 
    Uri uri = getIntent().getData(); 
    Cursor intentCursor = this.getContentResolver().query(uri, null, null, null, null); 
    Contact contact = null; 
    if (intentCursor != null) { 
     intentCursor.moveToFirst(); 
     Uri lookupUri = LocalContactAsync.getLookupUri(intentCursor); 
     Cursor lookupCursor = this.getContentResolver().query(lookupUri, null, null, null, null); 
     contact = LocalContactAsync.getContacts(lookupCursor, this.getContentResolver()).get(0); 
     intentCursor.close(); 
     lookupCursor.close(); 
    } 
} 

Контакт теперь будет содержать номера телефонов, как требуется.

+0

Спасибо за информацию, но я не начинаю действие для результата. Я добавляю настраиваемое событие к контактному лицу, которое открывает мое приложение непосредственно из контактного приложения Android. Как мне управлять этими дополнительными услугами? – jobbert

+0

Я добавил дополнительную информацию к вопросу. Код, который вы предоставили, работал так же, как и мой. Я получаю имя и идентификатор, но не номера телефонов. В то время как Contact имеет 3 телефонных номера. Также, если я просто получаю все контакты, я могу получить номера телефонов. – jobbert

+0

Затем возвращается номер 2849. Это плохое имя. Это не асинхронно. Я могу опубликовать полный код, если вы хотите. – jobbert

Смежные вопросы