2015-06-09 14:37:20 +00:00
|
|
|
package org.thoughtcrime.securesms.database;
|
|
|
|
|
|
|
|
import android.content.ContentValues;
|
|
|
|
import android.content.Context;
|
|
|
|
import android.database.Cursor;
|
|
|
|
import android.database.sqlite.SQLiteDatabase;
|
|
|
|
import android.database.sqlite.SQLiteOpenHelper;
|
|
|
|
import android.net.Uri;
|
|
|
|
import android.support.annotation.NonNull;
|
|
|
|
import android.support.annotation.Nullable;
|
|
|
|
import android.util.Log;
|
|
|
|
|
2015-06-30 16:16:05 +00:00
|
|
|
import org.thoughtcrime.securesms.color.MaterialColor;
|
2016-08-26 23:53:23 +00:00
|
|
|
import org.thoughtcrime.securesms.recipients.RecipientFactory;
|
2015-06-09 14:37:20 +00:00
|
|
|
import org.thoughtcrime.securesms.recipients.Recipients;
|
|
|
|
import org.thoughtcrime.securesms.util.Util;
|
2016-03-23 17:34:41 +00:00
|
|
|
import org.whispersystems.libsignal.util.guava.Optional;
|
2015-06-09 14:37:20 +00:00
|
|
|
|
|
|
|
import java.util.Arrays;
|
|
|
|
|
|
|
|
|
|
|
|
public class RecipientPreferenceDatabase extends Database {
|
|
|
|
|
|
|
|
private static final String TAG = RecipientPreferenceDatabase.class.getSimpleName();
|
2015-06-11 20:00:50 +00:00
|
|
|
private static final String RECIPIENT_PREFERENCES_URI = "content://textsecure/recipients/";
|
2015-06-09 14:37:20 +00:00
|
|
|
|
2016-02-06 00:10:33 +00:00
|
|
|
private static final String TABLE_NAME = "recipient_preferences";
|
|
|
|
private static final String ID = "_id";
|
|
|
|
private static final String RECIPIENT_IDS = "recipient_ids";
|
|
|
|
private static final String BLOCK = "block";
|
|
|
|
private static final String NOTIFICATION = "notification";
|
|
|
|
private static final String VIBRATE = "vibrate";
|
|
|
|
private static final String MUTE_UNTIL = "mute_until";
|
|
|
|
private static final String COLOR = "color";
|
|
|
|
private static final String SEEN_INVITE_REMINDER = "seen_invite_reminder";
|
|
|
|
private static final String DEFAULT_SUBSCRIPTION_ID = "default_subscription_id";
|
2016-08-16 03:23:56 +00:00
|
|
|
private static final String EXPIRE_MESSAGES = "expire_messages";
|
2015-06-09 14:37:20 +00:00
|
|
|
|
|
|
|
public enum VibrateState {
|
|
|
|
DEFAULT(0), ENABLED(1), DISABLED(2);
|
|
|
|
|
|
|
|
private final int id;
|
|
|
|
|
|
|
|
VibrateState(int id) {
|
|
|
|
this.id = id;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getId() {
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static VibrateState fromId(int id) {
|
|
|
|
return values()[id];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static final String CREATE_TABLE =
|
|
|
|
"CREATE TABLE " + TABLE_NAME +
|
|
|
|
" (" + ID + " INTEGER PRIMARY KEY, " +
|
|
|
|
RECIPIENT_IDS + " TEXT UNIQUE, " +
|
|
|
|
BLOCK + " INTEGER DEFAULT 0," +
|
|
|
|
NOTIFICATION + " TEXT DEFAULT NULL, " +
|
|
|
|
VIBRATE + " INTEGER DEFAULT " + VibrateState.DEFAULT.getId() + ", " +
|
2015-06-23 22:10:50 +00:00
|
|
|
MUTE_UNTIL + " INTEGER DEFAULT 0, " +
|
2015-10-14 04:44:01 +00:00
|
|
|
COLOR + " TEXT DEFAULT NULL, " +
|
2016-02-06 00:10:33 +00:00
|
|
|
SEEN_INVITE_REMINDER + " INTEGER DEFAULT 0, " +
|
2016-08-16 03:23:56 +00:00
|
|
|
DEFAULT_SUBSCRIPTION_ID + " INTEGER DEFAULT -1, " +
|
|
|
|
EXPIRE_MESSAGES + " INTEGER DEFAULT 0);";
|
2015-06-09 14:37:20 +00:00
|
|
|
|
|
|
|
public RecipientPreferenceDatabase(Context context, SQLiteOpenHelper databaseHelper) {
|
|
|
|
super(context, databaseHelper);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Cursor getBlocked() {
|
|
|
|
SQLiteDatabase database = databaseHelper.getReadableDatabase();
|
|
|
|
|
2015-06-11 20:00:50 +00:00
|
|
|
Cursor cursor = database.query(TABLE_NAME, new String[] {ID, RECIPIENT_IDS}, BLOCK + " = 1",
|
|
|
|
null, null, null, null, null);
|
|
|
|
cursor.setNotificationUri(context.getContentResolver(), Uri.parse(RECIPIENT_PREFERENCES_URI));
|
|
|
|
|
|
|
|
return cursor;
|
2015-06-09 14:37:20 +00:00
|
|
|
}
|
|
|
|
|
2016-08-26 23:53:23 +00:00
|
|
|
public BlockedReader readerForBlocked(Cursor cursor) {
|
|
|
|
return new BlockedReader(context, cursor);
|
|
|
|
}
|
|
|
|
|
2015-06-09 14:37:20 +00:00
|
|
|
public Optional<RecipientsPreferences> getRecipientsPreferences(@NonNull long[] recipients) {
|
|
|
|
Arrays.sort(recipients);
|
|
|
|
|
|
|
|
SQLiteDatabase database = databaseHelper.getReadableDatabase();
|
|
|
|
Cursor cursor = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
cursor = database.query(TABLE_NAME, null, RECIPIENT_IDS + " = ?",
|
|
|
|
new String[] {Util.join(recipients, " ")},
|
|
|
|
null, null, null);
|
|
|
|
|
|
|
|
if (cursor != null && cursor.moveToNext()) {
|
2016-02-06 00:10:33 +00:00
|
|
|
boolean blocked = cursor.getInt(cursor.getColumnIndexOrThrow(BLOCK)) == 1;
|
|
|
|
String notification = cursor.getString(cursor.getColumnIndexOrThrow(NOTIFICATION));
|
|
|
|
int vibrateState = cursor.getInt(cursor.getColumnIndexOrThrow(VIBRATE));
|
|
|
|
long muteUntil = cursor.getLong(cursor.getColumnIndexOrThrow(MUTE_UNTIL));
|
|
|
|
String serializedColor = cursor.getString(cursor.getColumnIndexOrThrow(COLOR));
|
|
|
|
Uri notificationUri = notification == null ? null : Uri.parse(notification);
|
|
|
|
boolean seenInviteReminder = cursor.getInt(cursor.getColumnIndexOrThrow(SEEN_INVITE_REMINDER)) == 1;
|
|
|
|
int defaultSubscriptionId = cursor.getInt(cursor.getColumnIndexOrThrow(DEFAULT_SUBSCRIPTION_ID));
|
2016-08-16 03:23:56 +00:00
|
|
|
int expireMessages = cursor.getInt(cursor.getColumnIndexOrThrow(EXPIRE_MESSAGES));
|
2015-06-09 14:37:20 +00:00
|
|
|
|
2015-06-30 16:16:05 +00:00
|
|
|
MaterialColor color;
|
|
|
|
|
|
|
|
try {
|
|
|
|
color = serializedColor == null ? null : MaterialColor.fromSerialized(serializedColor);
|
|
|
|
} catch (MaterialColor.UnknownColorException e) {
|
|
|
|
Log.w(TAG, e);
|
|
|
|
color = null;
|
|
|
|
}
|
|
|
|
|
2015-06-09 14:37:20 +00:00
|
|
|
Log.w(TAG, "Muted until: " + muteUntil);
|
|
|
|
|
|
|
|
return Optional.of(new RecipientsPreferences(blocked, muteUntil,
|
|
|
|
VibrateState.fromId(vibrateState),
|
2016-02-06 00:10:33 +00:00
|
|
|
notificationUri, color, seenInviteReminder,
|
2016-08-16 03:23:56 +00:00
|
|
|
defaultSubscriptionId, expireMessages));
|
2015-06-09 14:37:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return Optional.absent();
|
|
|
|
} finally {
|
|
|
|
if (cursor != null) cursor.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-30 16:16:05 +00:00
|
|
|
public void setColor(Recipients recipients, MaterialColor color) {
|
2015-06-23 22:10:50 +00:00
|
|
|
ContentValues values = new ContentValues();
|
2015-06-30 16:16:05 +00:00
|
|
|
values.put(COLOR, color.serialize());
|
2015-06-23 22:10:50 +00:00
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
2016-02-06 00:10:33 +00:00
|
|
|
public void setDefaultSubscriptionId(@NonNull Recipients recipients, int defaultSubscriptionId) {
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
values.put(DEFAULT_SUBSCRIPTION_ID, defaultSubscriptionId);
|
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
2015-06-09 14:37:20 +00:00
|
|
|
public void setBlocked(Recipients recipients, boolean blocked) {
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
values.put(BLOCK, blocked ? 1 : 0);
|
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setRingtone(Recipients recipients, @Nullable Uri notification) {
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
values.put(NOTIFICATION, notification == null ? null : notification.toString());
|
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setVibrate(Recipients recipients, @NonNull VibrateState enabled) {
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
values.put(VIBRATE, enabled.getId());
|
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setMuted(Recipients recipients, long until) {
|
|
|
|
Log.w(TAG, "Setting muted until: " + until);
|
|
|
|
ContentValues values = new ContentValues();
|
|
|
|
values.put(MUTE_UNTIL, until);
|
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
2015-10-14 04:44:01 +00:00
|
|
|
public void setSeenInviteReminder(Recipients recipients, boolean seen) {
|
|
|
|
ContentValues values = new ContentValues(1);
|
|
|
|
values.put(SEEN_INVITE_REMINDER, seen ? 1 : 0);
|
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
2016-08-16 03:23:56 +00:00
|
|
|
public void setExpireMessages(Recipients recipients, int expiration) {
|
|
|
|
recipients.setExpireMessages(expiration);
|
|
|
|
|
|
|
|
ContentValues values = new ContentValues(1);
|
|
|
|
values.put(EXPIRE_MESSAGES, expiration);
|
|
|
|
updateOrInsert(recipients, values);
|
|
|
|
}
|
|
|
|
|
2015-06-09 14:37:20 +00:00
|
|
|
private void updateOrInsert(Recipients recipients, ContentValues contentValues) {
|
|
|
|
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
|
|
|
|
|
|
|
database.beginTransaction();
|
|
|
|
|
|
|
|
int updated = database.update(TABLE_NAME, contentValues, RECIPIENT_IDS + " = ?",
|
|
|
|
new String[] {String.valueOf(recipients.getSortedIdsString())});
|
|
|
|
|
|
|
|
if (updated < 1) {
|
|
|
|
contentValues.put(RECIPIENT_IDS, recipients.getSortedIdsString());
|
|
|
|
database.insert(TABLE_NAME, null, contentValues);
|
|
|
|
}
|
|
|
|
|
|
|
|
database.setTransactionSuccessful();
|
|
|
|
database.endTransaction();
|
2015-06-11 20:00:50 +00:00
|
|
|
|
|
|
|
context.getContentResolver().notifyChange(Uri.parse(RECIPIENT_PREFERENCES_URI), null);
|
2015-06-09 14:37:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static class RecipientsPreferences {
|
2015-06-30 16:16:05 +00:00
|
|
|
private final boolean blocked;
|
|
|
|
private final long muteUntil;
|
|
|
|
private final VibrateState vibrateState;
|
|
|
|
private final Uri notification;
|
|
|
|
private final MaterialColor color;
|
2015-10-14 04:44:01 +00:00
|
|
|
private final boolean seenInviteReminder;
|
2016-02-06 00:10:33 +00:00
|
|
|
private final int defaultSubscriptionId;
|
2016-08-16 03:23:56 +00:00
|
|
|
private final int expireMessages;
|
2015-06-30 16:16:05 +00:00
|
|
|
|
|
|
|
public RecipientsPreferences(boolean blocked, long muteUntil,
|
|
|
|
@NonNull VibrateState vibrateState,
|
|
|
|
@Nullable Uri notification,
|
2015-10-14 04:44:01 +00:00
|
|
|
@Nullable MaterialColor color,
|
2016-02-06 00:10:33 +00:00
|
|
|
boolean seenInviteReminder,
|
2016-08-16 03:23:56 +00:00
|
|
|
int defaultSubscriptionId,
|
|
|
|
int expireMessages)
|
2015-06-23 22:10:50 +00:00
|
|
|
{
|
2016-02-06 00:10:33 +00:00
|
|
|
this.blocked = blocked;
|
|
|
|
this.muteUntil = muteUntil;
|
|
|
|
this.vibrateState = vibrateState;
|
|
|
|
this.notification = notification;
|
|
|
|
this.color = color;
|
|
|
|
this.seenInviteReminder = seenInviteReminder;
|
|
|
|
this.defaultSubscriptionId = defaultSubscriptionId;
|
2016-08-16 03:23:56 +00:00
|
|
|
this.expireMessages = expireMessages;
|
2015-06-23 22:10:50 +00:00
|
|
|
}
|
|
|
|
|
2015-06-30 16:16:05 +00:00
|
|
|
public @Nullable MaterialColor getColor() {
|
2015-06-23 22:10:50 +00:00
|
|
|
return color;
|
2015-06-09 14:37:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isBlocked() {
|
|
|
|
return blocked;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long getMuteUntil() {
|
|
|
|
return muteUntil;
|
|
|
|
}
|
|
|
|
|
|
|
|
public @NonNull VibrateState getVibrateState() {
|
|
|
|
return vibrateState;
|
|
|
|
}
|
|
|
|
|
|
|
|
public @Nullable Uri getRingtone() {
|
|
|
|
return notification;
|
|
|
|
}
|
2015-10-14 04:44:01 +00:00
|
|
|
|
|
|
|
public boolean hasSeenInviteReminder() {
|
|
|
|
return seenInviteReminder;
|
|
|
|
}
|
2016-02-06 00:10:33 +00:00
|
|
|
|
|
|
|
public Optional<Integer> getDefaultSubscriptionId() {
|
|
|
|
return defaultSubscriptionId != -1 ? Optional.of(defaultSubscriptionId) : Optional.<Integer>absent();
|
|
|
|
}
|
2016-08-16 03:23:56 +00:00
|
|
|
|
|
|
|
public int getExpireMessages() {
|
|
|
|
return expireMessages;
|
|
|
|
}
|
2015-06-09 14:37:20 +00:00
|
|
|
}
|
2016-08-26 23:53:23 +00:00
|
|
|
|
|
|
|
public static class BlockedReader {
|
|
|
|
|
|
|
|
private final Context context;
|
|
|
|
private final Cursor cursor;
|
|
|
|
|
|
|
|
public BlockedReader(Context context, Cursor cursor) {
|
|
|
|
this.context = context;
|
|
|
|
this.cursor = cursor;
|
|
|
|
}
|
|
|
|
|
|
|
|
public @NonNull Recipients getCurrent() {
|
|
|
|
String recipientIds = cursor.getString(cursor.getColumnIndexOrThrow(RECIPIENT_IDS));
|
|
|
|
return RecipientFactory.getRecipientsForIds(context, recipientIds, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public @Nullable Recipients getNext() {
|
|
|
|
if (!cursor.moveToNext()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return getCurrent();
|
|
|
|
}
|
|
|
|
}
|
2015-06-09 14:37:20 +00:00
|
|
|
}
|