|
|
|
@ -22,7 +22,6 @@ import android.content.Context;
|
|
|
|
|
import android.content.OperationApplicationException;
|
|
|
|
|
import android.database.Cursor;
|
|
|
|
|
import android.database.CursorWrapper;
|
|
|
|
|
import android.database.MatrixCursor;
|
|
|
|
|
import android.net.Uri;
|
|
|
|
|
import android.os.Build;
|
|
|
|
|
import android.os.RemoteException;
|
|
|
|
@ -37,16 +36,15 @@ import android.util.Pair;
|
|
|
|
|
|
|
|
|
|
import org.thoughtcrime.securesms.R;
|
|
|
|
|
import org.whispersystems.libaxolotl.util.guava.Optional;
|
|
|
|
|
import org.whispersystems.textsecure.api.push.ContactTokenDetails;
|
|
|
|
|
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
|
|
|
|
import org.whispersystems.textsecure.api.util.PhoneNumberFormatter;
|
|
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
|
import java.util.HashMap;
|
|
|
|
|
import java.util.HashSet;
|
|
|
|
|
import java.util.LinkedList;
|
|
|
|
|
import java.util.List;
|
|
|
|
|
import java.util.Map;
|
|
|
|
|
import java.util.Set;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Database to supply all types of contacts that TextSecure needs to know about
|
|
|
|
@ -55,9 +53,10 @@ import java.util.Set;
|
|
|
|
|
*/
|
|
|
|
|
public class ContactsDatabase {
|
|
|
|
|
|
|
|
|
|
private static final String TAG = ContactsDatabase.class.getSimpleName();
|
|
|
|
|
private static final String MIME = "vnd.android.cursor.item/vnd.org.thoughtcrime.securesms.contact";
|
|
|
|
|
private static final String SYNC = "__TS";
|
|
|
|
|
private static final String TAG = ContactsDatabase.class.getSimpleName();
|
|
|
|
|
private static final String CONTACT_MIMETYPE = "vnd.android.cursor.item/vnd.org.thoughtcrime.securesms.contact";
|
|
|
|
|
private static final String CALL_MIMETYPE = "vnd.android.cursor.item/vnd.org.thoughtcrime.securesms.call";
|
|
|
|
|
private static final String SYNC = "__TS";
|
|
|
|
|
|
|
|
|
|
public static final String ID_COLUMN = "_id";
|
|
|
|
|
public static final String NAME_COLUMN = "name";
|
|
|
|
@ -78,54 +77,44 @@ public class ContactsDatabase {
|
|
|
|
|
|
|
|
|
|
public synchronized @NonNull List<String> setRegisteredUsers(@NonNull Account account,
|
|
|
|
|
@NonNull String localNumber,
|
|
|
|
|
@NonNull List<String> e164numbers)
|
|
|
|
|
@NonNull List<ContactTokenDetails> registeredContacts)
|
|
|
|
|
throws RemoteException, OperationApplicationException
|
|
|
|
|
{
|
|
|
|
|
Map<String, Long> currentContacts = new HashMap<>();
|
|
|
|
|
Set<String> registeredNumbers = new HashSet<>(e164numbers);
|
|
|
|
|
List<String> addedNumbers = new LinkedList<>();
|
|
|
|
|
ArrayList<ContentProviderOperation> operations = new ArrayList<>();
|
|
|
|
|
Uri currentContactsUri = RawContacts.CONTENT_URI.buildUpon()
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type).build();
|
|
|
|
|
|
|
|
|
|
Cursor cursor = null;
|
|
|
|
|
Map<String, ContactTokenDetails> registeredNumbers = new HashMap<>();
|
|
|
|
|
List<String> addedNumbers = new LinkedList<>();
|
|
|
|
|
ArrayList<ContentProviderOperation> operations = new ArrayList<>();
|
|
|
|
|
Map<String, SignalContact> currentContacts = getSignalRawContacts(account, localNumber);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
cursor = context.getContentResolver().query(currentContactsUri, new String[] {BaseColumns._ID, RawContacts.SYNC1}, null, null, null);
|
|
|
|
|
for (ContactTokenDetails registeredContact : registeredContacts) {
|
|
|
|
|
String registeredNumber = registeredContact.getNumber();
|
|
|
|
|
|
|
|
|
|
while (cursor != null && cursor.moveToNext()) {
|
|
|
|
|
String currentNumber;
|
|
|
|
|
registeredNumbers.put(registeredNumber, registeredContact);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
currentNumber = PhoneNumberFormatter.formatNumber(cursor.getString(1), localNumber);
|
|
|
|
|
} catch (InvalidNumberException e) {
|
|
|
|
|
Log.w(TAG, e);
|
|
|
|
|
currentNumber = cursor.getString(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
currentContacts.put(currentNumber, cursor.getLong(0));
|
|
|
|
|
}
|
|
|
|
|
} finally {
|
|
|
|
|
if (cursor != null)
|
|
|
|
|
cursor.close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (String number : e164numbers) {
|
|
|
|
|
if (!currentContacts.containsKey(number)) {
|
|
|
|
|
Optional<SystemContactInfo> systemContactInfo = getSystemContactInfo(number, localNumber);
|
|
|
|
|
if (!currentContacts.containsKey(registeredNumber)) {
|
|
|
|
|
Optional<SystemContactInfo> systemContactInfo = getSystemContactInfo(registeredNumber, localNumber);
|
|
|
|
|
|
|
|
|
|
if (systemContactInfo.isPresent()) {
|
|
|
|
|
Log.w(TAG, "Adding number: " + number);
|
|
|
|
|
addedNumbers.add(number);
|
|
|
|
|
addTextSecureRawContact(operations, account, systemContactInfo.get().number, systemContactInfo.get().id);
|
|
|
|
|
Log.w(TAG, "Adding number: " + registeredNumber);
|
|
|
|
|
addedNumbers.add(registeredNumber);
|
|
|
|
|
addTextSecureRawContact(operations, account, systemContactInfo.get().number,
|
|
|
|
|
systemContactInfo.get().id, registeredContact.isVoice());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (Map.Entry<String, Long> currentContactEntry : currentContacts.entrySet()) {
|
|
|
|
|
if (!registeredNumbers.contains(currentContactEntry.getKey())) {
|
|
|
|
|
removeTextSecureRawContact(operations, account, currentContactEntry.getValue());
|
|
|
|
|
for (Map.Entry<String, SignalContact> currentContactEntry : currentContacts.entrySet()) {
|
|
|
|
|
ContactTokenDetails tokenDetails = registeredNumbers.get(currentContactEntry.getKey());
|
|
|
|
|
|
|
|
|
|
if (tokenDetails == null) {
|
|
|
|
|
Log.w(TAG, "Removing number: " + currentContactEntry.getKey());
|
|
|
|
|
removeTextSecureRawContact(operations, account, currentContactEntry.getValue().getId());
|
|
|
|
|
} else if (tokenDetails.isVoice() && !currentContactEntry.getValue().isVoiceSupported()) {
|
|
|
|
|
Log.w(TAG, "Adding voice support: " + currentContactEntry.getKey());
|
|
|
|
|
addContactVoiceSupport(operations, currentContactEntry.getKey(), currentContactEntry.getValue().getId());
|
|
|
|
|
} else if (!tokenDetails.isVoice() && currentContactEntry.getValue().isVoiceSupported()) {
|
|
|
|
|
Log.w(TAG, "Removing voice support: " + currentContactEntry.getKey());
|
|
|
|
|
removeContactVoiceSupport(operations, currentContactEntry.getValue().getId());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -136,60 +125,6 @@ public class ContactsDatabase {
|
|
|
|
|
return addedNumbers;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void addTextSecureRawContact(List<ContentProviderOperation> operations,
|
|
|
|
|
Account account,
|
|
|
|
|
String e164number,
|
|
|
|
|
long aggregateId)
|
|
|
|
|
{
|
|
|
|
|
int index = operations.size();
|
|
|
|
|
Uri dataUri = ContactsContract.Data.CONTENT_URI.buildUpon()
|
|
|
|
|
.appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true")
|
|
|
|
|
.build();
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(RawContacts.CONTENT_URI)
|
|
|
|
|
.withValue(RawContacts.ACCOUNT_NAME, account.name)
|
|
|
|
|
.withValue(RawContacts.ACCOUNT_TYPE, account.type)
|
|
|
|
|
.withValue(RawContacts.SYNC1, e164number)
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(dataUri)
|
|
|
|
|
.withValueBackReference(ContactsContract.CommonDataKinds.Phone.RAW_CONTACT_ID, index)
|
|
|
|
|
.withValue(ContactsContract.Data.MIMETYPE, ContactsContract.CommonDataKinds.Phone.CONTENT_ITEM_TYPE)
|
|
|
|
|
.withValue(ContactsContract.CommonDataKinds.Phone.NUMBER, e164number)
|
|
|
|
|
.withValue(ContactsContract.CommonDataKinds.Phone.TYPE, ContactsContract.CommonDataKinds.Phone.TYPE_OTHER)
|
|
|
|
|
.withValue(ContactsContract.Data.SYNC2, SYNC)
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(dataUri)
|
|
|
|
|
.withValueBackReference(ContactsContract.Data.RAW_CONTACT_ID, index)
|
|
|
|
|
.withValue(ContactsContract.Data.MIMETYPE, MIME)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA1, e164number)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA2, context.getString(R.string.app_name))
|
|
|
|
|
.withValue(ContactsContract.Data.DATA3, context.getString(R.string.ContactsDatabase_message_s, e164number))
|
|
|
|
|
.withYieldAllowed(true)
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
if (Build.VERSION.SDK_INT >= 11) {
|
|
|
|
|
operations.add(ContentProviderOperation.newUpdate(ContactsContract.AggregationExceptions.CONTENT_URI)
|
|
|
|
|
.withValue(ContactsContract.AggregationExceptions.RAW_CONTACT_ID1, aggregateId)
|
|
|
|
|
.withValueBackReference(ContactsContract.AggregationExceptions.RAW_CONTACT_ID2, index)
|
|
|
|
|
.withValue(ContactsContract.AggregationExceptions.TYPE, ContactsContract.AggregationExceptions.TYPE_KEEP_TOGETHER)
|
|
|
|
|
.build());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void removeTextSecureRawContact(List<ContentProviderOperation> operations,
|
|
|
|
|
Account account, long rowId)
|
|
|
|
|
{
|
|
|
|
|
operations.add(ContentProviderOperation.newDelete(RawContacts.CONTENT_URI.buildUpon()
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type)
|
|
|
|
|
.appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true").build())
|
|
|
|
|
.withYieldAllowed(true)
|
|
|
|
|
.withSelection(BaseColumns._ID + " = ?", new String[] {String.valueOf(rowId)})
|
|
|
|
|
.build());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public @NonNull Cursor querySystemContacts(String filter) {
|
|
|
|
|
Uri uri;
|
|
|
|
|
|
|
|
|
@ -248,13 +183,13 @@ public class ContactsDatabase {
|
|
|
|
|
cursor = context.getContentResolver().query(ContactsContract.Data.CONTENT_URI,
|
|
|
|
|
projection,
|
|
|
|
|
ContactsContract.Data.MIMETYPE + " = ?",
|
|
|
|
|
new String[] {MIME},
|
|
|
|
|
new String[] {CONTACT_MIMETYPE},
|
|
|
|
|
sort);
|
|
|
|
|
} else {
|
|
|
|
|
cursor = context.getContentResolver().query(ContactsContract.Data.CONTENT_URI,
|
|
|
|
|
projection,
|
|
|
|
|
ContactsContract.Data.MIMETYPE + " = ? AND (" + ContactsContract.Contacts.DISPLAY_NAME + " LIKE ? OR " + ContactsContract.Data.DATA1 + " LIKE ?)",
|
|
|
|
|
new String[] {MIME,
|
|
|
|
|
new String[] {CONTACT_MIMETYPE,
|
|
|
|
|
"%" + filter + "%", "%" + filter + "%"},
|
|
|
|
|
sort);
|
|
|
|
|
}
|
|
|
|
@ -266,13 +201,132 @@ public class ContactsDatabase {
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Cursor getNewNumberCursor(String filter) {
|
|
|
|
|
MatrixCursor newNumberCursor = new MatrixCursor(new String[] {ID_COLUMN, NAME_COLUMN, NUMBER_COLUMN, NUMBER_TYPE_COLUMN, LABEL_COLUMN, CONTACT_TYPE_COLUMN}, 1);
|
|
|
|
|
newNumberCursor.addRow(new Object[]{-1L, context.getString(R.string.contact_selection_list__unknown_contact),
|
|
|
|
|
filter, ContactsContract.CommonDataKinds.Phone.TYPE_CUSTOM,
|
|
|
|
|
"\u21e2", NEW_TYPE});
|
|
|
|
|
private void addContactVoiceSupport(List<ContentProviderOperation> operations,
|
|
|
|
|
@NonNull String e164number, long rawContactId)
|
|
|
|
|
{
|
|
|
|
|
operations.add(ContentProviderOperation.newUpdate(RawContacts.CONTENT_URI)
|
|
|
|
|
.withSelection(RawContacts._ID + " = ?", new String[] {String.valueOf(rawContactId)})
|
|
|
|
|
.withValue(RawContacts.SYNC4, "true")
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(ContactsContract.Data.CONTENT_URI.buildUpon().appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true").build())
|
|
|
|
|
.withValue(ContactsContract.Data.RAW_CONTACT_ID, rawContactId)
|
|
|
|
|
.withValue(ContactsContract.Data.MIMETYPE, CALL_MIMETYPE)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA1, e164number)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA2, context.getString(R.string.app_name))
|
|
|
|
|
.withValue(ContactsContract.Data.DATA3, context.getString(R.string.ContactsDatabase_signal_call_s, e164number))
|
|
|
|
|
.withYieldAllowed(true)
|
|
|
|
|
.build());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void removeContactVoiceSupport(List<ContentProviderOperation> operations, long rawContactId) {
|
|
|
|
|
operations.add(ContentProviderOperation.newUpdate(RawContacts.CONTENT_URI)
|
|
|
|
|
.withSelection(RawContacts._ID + " = ?", new String[] {String.valueOf(rawContactId)})
|
|
|
|
|
.withValue(RawContacts.SYNC4, "false")
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newDelete(ContactsContract.Data.CONTENT_URI.buildUpon().appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true").build())
|
|
|
|
|
.withSelection(ContactsContract.Data.RAW_CONTACT_ID + " = ? AND " + ContactsContract.Data.MIMETYPE + " = ?",
|
|
|
|
|
new String[] {String.valueOf(rawContactId), CALL_MIMETYPE})
|
|
|
|
|
.withYieldAllowed(true)
|
|
|
|
|
.build());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void addTextSecureRawContact(List<ContentProviderOperation> operations,
|
|
|
|
|
Account account, String e164number,
|
|
|
|
|
long aggregateId, boolean supportsVoice)
|
|
|
|
|
{
|
|
|
|
|
int index = operations.size();
|
|
|
|
|
Uri dataUri = ContactsContract.Data.CONTENT_URI.buildUpon()
|
|
|
|
|
.appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true")
|
|
|
|
|
.build();
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(RawContacts.CONTENT_URI)
|
|
|
|
|
.withValue(RawContacts.ACCOUNT_NAME, account.name)
|
|
|
|
|
.withValue(RawContacts.ACCOUNT_TYPE, account.type)
|
|
|
|
|
.withValue(RawContacts.SYNC1, e164number)
|
|
|
|
|
.withValue(RawContacts.SYNC4, String.valueOf(supportsVoice))
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(dataUri)
|
|
|
|
|
.withValueBackReference(ContactsContract.CommonDataKinds.Phone.RAW_CONTACT_ID, index)
|
|
|
|
|
.withValue(ContactsContract.Data.MIMETYPE, ContactsContract.CommonDataKinds.Phone.CONTENT_ITEM_TYPE)
|
|
|
|
|
.withValue(ContactsContract.CommonDataKinds.Phone.NUMBER, e164number)
|
|
|
|
|
.withValue(ContactsContract.CommonDataKinds.Phone.TYPE, ContactsContract.CommonDataKinds.Phone.TYPE_OTHER)
|
|
|
|
|
.withValue(ContactsContract.Data.SYNC2, SYNC)
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(dataUri)
|
|
|
|
|
.withValueBackReference(ContactsContract.Data.RAW_CONTACT_ID, index)
|
|
|
|
|
.withValue(ContactsContract.Data.MIMETYPE, CONTACT_MIMETYPE)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA1, e164number)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA2, context.getString(R.string.app_name))
|
|
|
|
|
.withValue(ContactsContract.Data.DATA3, context.getString(R.string.ContactsDatabase_message_s, e164number))
|
|
|
|
|
.withYieldAllowed(true)
|
|
|
|
|
.build());
|
|
|
|
|
|
|
|
|
|
if (supportsVoice) {
|
|
|
|
|
operations.add(ContentProviderOperation.newInsert(dataUri)
|
|
|
|
|
.withValueBackReference(ContactsContract.Data.RAW_CONTACT_ID, index)
|
|
|
|
|
.withValue(ContactsContract.Data.MIMETYPE, CALL_MIMETYPE)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA1, e164number)
|
|
|
|
|
.withValue(ContactsContract.Data.DATA2, context.getString(R.string.app_name))
|
|
|
|
|
.withValue(ContactsContract.Data.DATA3, context.getString(R.string.ContactsDatabase_signal_call_s, e164number))
|
|
|
|
|
.withYieldAllowed(true)
|
|
|
|
|
.build());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (Build.VERSION.SDK_INT >= 11) {
|
|
|
|
|
operations.add(ContentProviderOperation.newUpdate(ContactsContract.AggregationExceptions.CONTENT_URI)
|
|
|
|
|
.withValue(ContactsContract.AggregationExceptions.RAW_CONTACT_ID1, aggregateId)
|
|
|
|
|
.withValueBackReference(ContactsContract.AggregationExceptions.RAW_CONTACT_ID2, index)
|
|
|
|
|
.withValue(ContactsContract.AggregationExceptions.TYPE, ContactsContract.AggregationExceptions.TYPE_KEEP_TOGETHER)
|
|
|
|
|
.build());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void removeTextSecureRawContact(List<ContentProviderOperation> operations,
|
|
|
|
|
Account account, long rowId)
|
|
|
|
|
{
|
|
|
|
|
operations.add(ContentProviderOperation.newDelete(RawContacts.CONTENT_URI.buildUpon()
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type)
|
|
|
|
|
.appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true").build())
|
|
|
|
|
.withYieldAllowed(true)
|
|
|
|
|
.withSelection(BaseColumns._ID + " = ?", new String[] {String.valueOf(rowId)})
|
|
|
|
|
.build());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private @NonNull Map<String, SignalContact> getSignalRawContacts(Account account, String localNumber) {
|
|
|
|
|
Uri currentContactsUri = RawContacts.CONTENT_URI.buildUpon()
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
|
|
|
|
|
.appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type).build();
|
|
|
|
|
|
|
|
|
|
Map<String, SignalContact> signalContacts = new HashMap<>();
|
|
|
|
|
Cursor cursor = null;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
cursor = context.getContentResolver().query(currentContactsUri, new String[] {BaseColumns._ID, RawContacts.SYNC1, RawContacts.SYNC4}, null, null, null);
|
|
|
|
|
|
|
|
|
|
while (cursor != null && cursor.moveToNext()) {
|
|
|
|
|
String currentNumber;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
currentNumber = PhoneNumberFormatter.formatNumber(cursor.getString(1), localNumber);
|
|
|
|
|
} catch (InvalidNumberException e) {
|
|
|
|
|
Log.w(TAG, e);
|
|
|
|
|
currentNumber = cursor.getString(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
signalContacts.put(currentNumber, new SignalContact(cursor.getLong(0), cursor.getString(2)));
|
|
|
|
|
}
|
|
|
|
|
} finally {
|
|
|
|
|
if (cursor != null)
|
|
|
|
|
cursor.close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return newNumberCursor;
|
|
|
|
|
return signalContacts;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Optional<SystemContactInfo> getSystemContactInfo(@NonNull String e164number,
|
|
|
|
@ -428,4 +482,22 @@ public class ContactsDatabase {
|
|
|
|
|
this.id = id;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static class SignalContact {
|
|
|
|
|
private final long id;
|
|
|
|
|
@Nullable private final String supportsVoice;
|
|
|
|
|
|
|
|
|
|
public SignalContact(long id, @Nullable String supportsVoice) {
|
|
|
|
|
this.id = id;
|
|
|
|
|
this.supportsVoice = supportsVoice;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public long getId() {
|
|
|
|
|
return id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public boolean isVoiceSupported() {
|
|
|
|
|
return "true".equals(supportsVoice);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|