2012-09-11 18:23:19 -07:00
|
|
|
/**
|
2011-12-20 10:20:44 -08:00
|
|
|
* Copyright (C) 2011 Whisper Systems
|
2012-09-11 18:23:19 -07:00
|
|
|
*
|
2011-12-20 10:20:44 -08:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2012-09-11 18:23:19 -07:00
|
|
|
*
|
2011-12-20 10:20:44 -08:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
package org.thoughtcrime.securesms.recipients;
|
|
|
|
|
|
|
|
import android.content.Context;
|
2012-12-12 03:56:38 -08:00
|
|
|
import android.database.Cursor;
|
2011-12-20 10:20:44 -08:00
|
|
|
import android.net.Uri;
|
2012-12-12 03:56:38 -08:00
|
|
|
import android.provider.ContactsContract.Contacts;
|
|
|
|
import android.provider.ContactsContract.PhoneLookup;
|
2015-06-23 10:15:33 -07:00
|
|
|
import android.support.annotation.NonNull;
|
2015-06-09 07:37:20 -07:00
|
|
|
import android.support.annotation.Nullable;
|
2017-08-04 09:28:20 -07:00
|
|
|
import android.text.TextUtils;
|
2012-12-24 08:40:37 -08:00
|
|
|
import android.util.Log;
|
2011-12-20 10:20:44 -08:00
|
|
|
|
2015-07-27 10:49:14 -07:00
|
|
|
import org.thoughtcrime.securesms.R;
|
2015-06-30 09:16:05 -07:00
|
|
|
import org.thoughtcrime.securesms.color.MaterialColor;
|
2015-06-23 10:15:33 -07:00
|
|
|
import org.thoughtcrime.securesms.contacts.avatars.ContactPhoto;
|
|
|
|
import org.thoughtcrime.securesms.contacts.avatars.ContactPhotoFactory;
|
2017-07-26 09:59:15 -07:00
|
|
|
import org.thoughtcrime.securesms.database.Address;
|
2014-01-14 00:26:43 -08:00
|
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
2017-08-07 16:47:38 -07:00
|
|
|
import org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
2015-06-09 07:37:20 -07:00
|
|
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase.RecipientsPreferences;
|
2017-08-01 08:56:00 -07:00
|
|
|
import org.thoughtcrime.securesms.database.RecipientPreferenceDatabase.VibrateState;
|
2012-12-24 08:40:37 -08:00
|
|
|
import org.thoughtcrime.securesms.util.LRUCache;
|
2014-11-12 09:20:57 -08:00
|
|
|
import org.thoughtcrime.securesms.util.ListenableFutureTask;
|
2015-05-04 11:36:18 -07:00
|
|
|
import org.thoughtcrime.securesms.util.Util;
|
2016-03-23 10:34:41 -07:00
|
|
|
import org.whispersystems.libsignal.util.guava.Optional;
|
2012-09-11 18:23:19 -07:00
|
|
|
|
2015-07-27 10:49:14 -07:00
|
|
|
import java.util.HashMap;
|
2015-06-09 07:37:20 -07:00
|
|
|
import java.util.LinkedList;
|
|
|
|
import java.util.List;
|
2012-12-24 08:40:37 -08:00
|
|
|
import java.util.Map;
|
2012-12-27 12:00:14 -08:00
|
|
|
import java.util.concurrent.Callable;
|
|
|
|
import java.util.concurrent.ExecutorService;
|
2012-12-12 03:56:38 -08:00
|
|
|
|
2017-01-22 21:57:23 -08:00
|
|
|
class RecipientProvider {
|
2012-09-11 18:23:19 -07:00
|
|
|
|
2015-06-23 10:15:33 -07:00
|
|
|
private static final String TAG = RecipientProvider.class.getSimpleName();
|
|
|
|
|
2015-07-17 10:16:14 -07:00
|
|
|
private static final RecipientCache recipientCache = new RecipientCache();
|
|
|
|
private static final ExecutorService asyncRecipientResolver = Util.newSingleThreadedLifoExecutor();
|
2012-09-11 18:23:19 -07:00
|
|
|
|
2012-12-12 03:56:38 -08:00
|
|
|
private static final String[] CALLER_ID_PROJECTION = new String[] {
|
|
|
|
PhoneLookup.DISPLAY_NAME,
|
|
|
|
PhoneLookup.LOOKUP_KEY,
|
|
|
|
PhoneLookup._ID,
|
2017-02-20 16:48:55 +01:00
|
|
|
PhoneLookup.NUMBER,
|
|
|
|
PhoneLookup.LABEL
|
2012-12-12 03:56:38 -08:00
|
|
|
};
|
|
|
|
|
2015-07-27 10:49:14 -07:00
|
|
|
private static final Map<String, RecipientDetails> STATIC_DETAILS = new HashMap<String, RecipientDetails>() {{
|
2017-07-26 09:59:15 -07:00
|
|
|
put("262966", new RecipientDetails("Amazon", null, null,
|
2015-07-27 10:49:14 -07:00
|
|
|
ContactPhotoFactory.getResourceContactPhoto(R.drawable.ic_amazon),
|
2017-08-01 08:56:00 -07:00
|
|
|
null, null));
|
2015-07-27 10:49:14 -07:00
|
|
|
}};
|
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
@NonNull Recipient getRecipient(Context context, Address address, Optional<RecipientsPreferences> preferences, Optional<GroupRecord> groupRecord, boolean asynchronous) {
|
2017-07-26 09:59:15 -07:00
|
|
|
Recipient cachedRecipient = recipientCache.get(address);
|
2017-01-22 21:57:23 -08:00
|
|
|
if (cachedRecipient != null && !cachedRecipient.isStale() && (asynchronous || !cachedRecipient.isResolving())) {
|
|
|
|
return cachedRecipient;
|
|
|
|
}
|
2015-06-23 10:15:33 -07:00
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
Optional<RecipientDetails> prefetchedRecipientDetails = createPrefetchedRecipientDetails(context, address, preferences, groupRecord);
|
|
|
|
|
2015-06-23 10:15:33 -07:00
|
|
|
if (asynchronous) {
|
2017-08-07 16:47:38 -07:00
|
|
|
cachedRecipient = new Recipient(address, cachedRecipient, prefetchedRecipientDetails, getRecipientDetailsAsync(context, address, preferences, groupRecord));
|
2015-06-23 10:15:33 -07:00
|
|
|
} else {
|
2017-08-07 16:47:38 -07:00
|
|
|
cachedRecipient = new Recipient(address, getRecipientDetailsSync(context, address, preferences, groupRecord, false));
|
2015-06-23 10:15:33 -07:00
|
|
|
}
|
2012-12-12 03:56:38 -08:00
|
|
|
|
2017-07-26 09:59:15 -07:00
|
|
|
recipientCache.set(address, cachedRecipient);
|
2015-06-23 10:15:33 -07:00
|
|
|
return cachedRecipient;
|
2015-06-09 07:37:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-23 10:15:33 -07:00
|
|
|
void clearCache() {
|
2015-07-17 10:16:14 -07:00
|
|
|
recipientCache.reset();
|
2014-01-14 00:26:43 -08:00
|
|
|
}
|
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
private @NonNull Optional<RecipientDetails> createPrefetchedRecipientDetails(@NonNull Context context, @NonNull Address address,
|
|
|
|
@NonNull Optional<RecipientsPreferences> preferences,
|
|
|
|
@NonNull Optional<GroupRecord> groupRecord)
|
|
|
|
{
|
|
|
|
if (address.isGroup() && preferences.isPresent() && groupRecord.isPresent()) {
|
|
|
|
return Optional.of(getGroupRecipientDetails(context, address, groupRecord, preferences, true));
|
|
|
|
} else if (!address.isGroup() && preferences.isPresent()) {
|
|
|
|
return Optional.of(new RecipientDetails(null, null, null, ContactPhotoFactory.getLoadingPhoto(), preferences.get(), null));
|
|
|
|
}
|
|
|
|
|
|
|
|
return Optional.absent();
|
|
|
|
}
|
|
|
|
|
|
|
|
private @NonNull ListenableFutureTask<RecipientDetails> getRecipientDetailsAsync(final Context context, final @NonNull Address address, final @NonNull Optional<RecipientsPreferences> preferences, final @NonNull Optional<GroupRecord> groupRecord)
|
2015-06-23 10:15:33 -07:00
|
|
|
{
|
2014-01-14 00:26:43 -08:00
|
|
|
Callable<RecipientDetails> task = new Callable<RecipientDetails>() {
|
|
|
|
@Override
|
|
|
|
public RecipientDetails call() throws Exception {
|
2017-08-07 16:47:38 -07:00
|
|
|
return getRecipientDetailsSync(context, address, preferences, groupRecord, true);
|
2014-01-14 00:26:43 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-05-04 11:36:18 -07:00
|
|
|
ListenableFutureTask<RecipientDetails> future = new ListenableFutureTask<>(task);
|
2014-01-14 00:26:43 -08:00
|
|
|
asyncRecipientResolver.submit(future);
|
2015-06-23 10:15:33 -07:00
|
|
|
return future;
|
2012-12-24 08:40:37 -08:00
|
|
|
}
|
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
private @NonNull RecipientDetails getRecipientDetailsSync(Context context, @NonNull Address address, Optional<RecipientsPreferences> preferences, Optional<GroupRecord> groupRecord, boolean nestedAsynchronous) {
|
|
|
|
if (address.isGroup()) return getGroupRecipientDetails(context, address, groupRecord, preferences, nestedAsynchronous);
|
2017-08-06 21:43:11 -07:00
|
|
|
else return getIndividualRecipientDetails(context, address, preferences);
|
2014-02-20 15:41:52 -08:00
|
|
|
}
|
|
|
|
|
2017-08-06 21:43:11 -07:00
|
|
|
private @NonNull RecipientDetails getIndividualRecipientDetails(Context context, @NonNull Address address, Optional<RecipientsPreferences> preferences) {
|
|
|
|
if (!preferences.isPresent()) {
|
|
|
|
preferences = DatabaseFactory.getRecipientPreferenceDatabase(context).getRecipientsPreferences(address);
|
|
|
|
}
|
2017-08-01 09:57:50 -07:00
|
|
|
|
2017-08-04 09:28:20 -07:00
|
|
|
if (address.isPhone() && !TextUtils.isEmpty(address.toPhoneString())) {
|
2017-08-01 09:57:50 -07:00
|
|
|
Uri uri = Uri.withAppendedPath(PhoneLookup.CONTENT_FILTER_URI, Uri.encode(address.toPhoneString()));
|
|
|
|
Cursor cursor = context.getContentResolver().query(uri, CALLER_ID_PROJECTION, null, null, null);
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (cursor != null && cursor.moveToFirst()) {
|
|
|
|
final String resultNumber = cursor.getString(3);
|
|
|
|
if (resultNumber != null) {
|
|
|
|
Uri contactUri = Contacts.getLookupUri(cursor.getLong(2), cursor.getString(1));
|
|
|
|
String name = resultNumber.equals(cursor.getString(0)) ? null : cursor.getString(0);
|
|
|
|
ContactPhoto contactPhoto = ContactPhotoFactory.getContactPhoto(context,
|
|
|
|
Uri.withAppendedPath(Contacts.CONTENT_URI, cursor.getLong(2) + ""),
|
2017-08-14 18:11:13 -07:00
|
|
|
address,
|
2017-08-01 09:57:50 -07:00
|
|
|
name);
|
|
|
|
|
2017-08-01 08:56:00 -07:00
|
|
|
return new RecipientDetails(cursor.getString(0), cursor.getString(4), contactUri, contactPhoto, preferences.orNull(), null);
|
2017-08-01 09:57:50 -07:00
|
|
|
} else {
|
|
|
|
Log.w(TAG, "resultNumber is null");
|
|
|
|
}
|
2016-04-24 17:56:03 +02:00
|
|
|
}
|
2017-08-01 09:57:50 -07:00
|
|
|
} finally {
|
|
|
|
if (cursor != null)
|
|
|
|
cursor.close();
|
2012-12-12 03:56:38 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-01 09:57:50 -07:00
|
|
|
if (STATIC_DETAILS.containsKey(address.serialize())) return STATIC_DETAILS.get(address.serialize());
|
2017-08-01 08:56:00 -07:00
|
|
|
else return new RecipientDetails(null, null, null, ContactPhotoFactory.getDefaultContactPhoto(null), preferences.orNull(), null);
|
2012-12-12 03:56:38 -08:00
|
|
|
}
|
2011-12-20 10:20:44 -08:00
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
private @NonNull RecipientDetails getGroupRecipientDetails(Context context, Address groupId, Optional<GroupRecord> groupRecord, Optional<RecipientsPreferences> preferences, boolean asynchronous) {
|
|
|
|
if (!groupRecord.isPresent()) {
|
|
|
|
groupRecord = DatabaseFactory.getGroupDatabase(context).getGroup(groupId.toGroupString());
|
|
|
|
}
|
2014-02-02 19:38:06 -08:00
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
if (!preferences.isPresent()) {
|
|
|
|
preferences = DatabaseFactory.getRecipientPreferenceDatabase(context).getRecipientsPreferences(groupId);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (groupRecord.isPresent()) {
|
|
|
|
ContactPhoto contactPhoto = ContactPhotoFactory.getGroupContactPhoto(groupRecord.get().getAvatar());
|
|
|
|
String title = groupRecord.get().getTitle();
|
|
|
|
List<Address> memberAddresses = groupRecord.get().getMembers();
|
2017-08-01 08:56:00 -07:00
|
|
|
List<Recipient> members = new LinkedList<>();
|
2017-01-22 21:23:51 -08:00
|
|
|
|
2017-08-01 08:56:00 -07:00
|
|
|
for (Address memberAddress : memberAddresses) {
|
2017-08-07 16:47:38 -07:00
|
|
|
members.add(getRecipient(context, memberAddress, Optional.absent(), Optional.absent(), asynchronous));
|
2014-01-14 00:26:43 -08:00
|
|
|
}
|
|
|
|
|
2017-08-01 08:56:00 -07:00
|
|
|
if (!groupId.isMmsGroup() && title == null) {
|
|
|
|
title = context.getString(R.string.RecipientProvider_unnamed_group);;
|
2015-06-09 07:37:20 -07:00
|
|
|
}
|
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
return new RecipientDetails(title, null, null, contactPhoto, preferences.orNull(), members);
|
2017-08-01 08:56:00 -07:00
|
|
|
}
|
2015-06-09 07:37:20 -07:00
|
|
|
|
2017-08-07 16:47:38 -07:00
|
|
|
return new RecipientDetails(context.getString(R.string.RecipientProvider_unnamed_group), null, null, ContactPhotoFactory.getDefaultGroupPhoto(), preferences.orNull(), null);
|
2015-06-09 07:37:20 -07:00
|
|
|
}
|
|
|
|
|
2017-08-01 08:56:00 -07:00
|
|
|
static class RecipientDetails {
|
|
|
|
@Nullable public final String name;
|
|
|
|
@Nullable public final String customLabel;
|
|
|
|
@NonNull public final ContactPhoto avatar;
|
|
|
|
@Nullable public final Uri contactUri;
|
|
|
|
@Nullable public final MaterialColor color;
|
|
|
|
@Nullable public final Uri ringtone;
|
|
|
|
public final long mutedUntil;
|
|
|
|
@Nullable public final VibrateState vibrateState;
|
|
|
|
public final boolean blocked;
|
|
|
|
public final int expireMessages;
|
|
|
|
@NonNull public final List<Recipient> participants;
|
2015-06-23 15:10:50 -07:00
|
|
|
|
2017-07-26 09:59:15 -07:00
|
|
|
public RecipientDetails(@Nullable String name, @Nullable String customLabel,
|
|
|
|
@Nullable Uri contactUri, @NonNull ContactPhoto avatar,
|
2017-08-01 08:56:00 -07:00
|
|
|
@Nullable RecipientsPreferences preferences,
|
|
|
|
@Nullable List<Recipient> participants)
|
2015-06-23 15:10:50 -07:00
|
|
|
{
|
2017-08-01 08:56:00 -07:00
|
|
|
this.customLabel = customLabel;
|
|
|
|
this.avatar = avatar;
|
|
|
|
this.contactUri = contactUri;
|
|
|
|
this.color = preferences != null ? preferences.getColor() : null;
|
|
|
|
this.ringtone = preferences != null ? preferences.getRingtone() : null;
|
|
|
|
this.mutedUntil = preferences != null ? preferences.getMuteUntil() : 0;
|
|
|
|
this.vibrateState = preferences != null ? preferences.getVibrateState() : null;
|
|
|
|
this.blocked = preferences != null && preferences.isBlocked();
|
|
|
|
this.expireMessages = preferences != null ? preferences.getExpireMessages() : 0;
|
|
|
|
this.participants = participants == null ? new LinkedList<Recipient>() : participants;
|
2017-08-07 16:47:38 -07:00
|
|
|
|
|
|
|
if (name == null && preferences != null) this.name = preferences.getSystemDisplayName();
|
|
|
|
else this.name = name;
|
2015-06-09 07:37:20 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-17 10:16:14 -07:00
|
|
|
private static class RecipientCache {
|
|
|
|
|
2017-07-26 09:59:15 -07:00
|
|
|
private final Map<Address,Recipient> cache = new LRUCache<>(1000);
|
2015-07-17 10:16:14 -07:00
|
|
|
|
2017-07-26 09:59:15 -07:00
|
|
|
public synchronized Recipient get(Address address) {
|
|
|
|
return cache.get(address);
|
2015-07-17 10:16:14 -07:00
|
|
|
}
|
|
|
|
|
2017-07-26 09:59:15 -07:00
|
|
|
public synchronized void set(Address address, Recipient recipient) {
|
|
|
|
cache.put(address, recipient);
|
2015-07-17 10:16:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
public synchronized void reset() {
|
|
|
|
for (Recipient recipient : cache.values()) {
|
|
|
|
recipient.setStale();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-12-20 10:20:44 -08:00
|
|
|
}
|