mirror of
https://github.com/oxen-io/session-android.git
synced 2024-11-25 11:05:25 +00:00
9f04c28bfd
You can now search for messages within a specific conversation.
450 lines
18 KiB
Java
450 lines
18 KiB
Java
/*
|
|
* Copyright (C) 2015 Open Whisper Systems
|
|
*
|
|
* 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.
|
|
*
|
|
* 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;
|
|
|
|
import android.annotation.SuppressLint;
|
|
import android.content.ClipData;
|
|
import android.content.ClipboardManager;
|
|
import android.content.Context;
|
|
import android.database.Cursor;
|
|
import android.graphics.drawable.ColorDrawable;
|
|
import android.os.AsyncTask;
|
|
import android.os.Build;
|
|
import android.os.Bundle;
|
|
import android.support.annotation.NonNull;
|
|
import android.support.annotation.Nullable;
|
|
import android.support.v4.app.LoaderManager.LoaderCallbacks;
|
|
import android.support.v4.content.Loader;
|
|
|
|
import org.thoughtcrime.securesms.conversation.ConversationItem;
|
|
import org.thoughtcrime.securesms.logging.Log;
|
|
import android.view.LayoutInflater;
|
|
import android.view.MenuItem;
|
|
import android.view.View;
|
|
import android.view.ViewGroup;
|
|
import android.widget.ListView;
|
|
import android.widget.TextView;
|
|
|
|
import org.thoughtcrime.securesms.MessageDetailsRecipientAdapter.RecipientDeliveryStatus;
|
|
import org.thoughtcrime.securesms.color.MaterialColor;
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
import org.thoughtcrime.securesms.database.GroupReceiptDatabase;
|
|
import org.thoughtcrime.securesms.database.GroupReceiptDatabase.GroupReceiptInfo;
|
|
import org.thoughtcrime.securesms.database.MmsDatabase;
|
|
import org.thoughtcrime.securesms.database.MmsSmsDatabase;
|
|
import org.thoughtcrime.securesms.database.SmsDatabase;
|
|
import org.thoughtcrime.securesms.database.loaders.MessageDetailsLoader;
|
|
import org.thoughtcrime.securesms.database.model.MessageRecord;
|
|
import org.thoughtcrime.securesms.mms.GlideApp;
|
|
import org.thoughtcrime.securesms.mms.GlideRequests;
|
|
import org.thoughtcrime.securesms.notifications.MessageNotifier;
|
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
|
import org.thoughtcrime.securesms.recipients.RecipientModifiedListener;
|
|
import org.thoughtcrime.securesms.sms.MessageSender;
|
|
import org.thoughtcrime.securesms.util.DateUtils;
|
|
import org.thoughtcrime.securesms.util.DynamicLanguage;
|
|
import org.thoughtcrime.securesms.util.DynamicTheme;
|
|
import org.thoughtcrime.securesms.util.ExpirationUtil;
|
|
import org.thoughtcrime.securesms.util.Util;
|
|
import org.whispersystems.libsignal.util.guava.Optional;
|
|
|
|
import java.lang.ref.WeakReference;
|
|
import java.sql.Date;
|
|
import java.text.SimpleDateFormat;
|
|
import java.util.HashSet;
|
|
import java.util.LinkedList;
|
|
import java.util.List;
|
|
import java.util.Locale;
|
|
|
|
/**
|
|
* @author Jake McGinty
|
|
*/
|
|
public class MessageDetailsActivity extends PassphraseRequiredActionBarActivity implements LoaderCallbacks<Cursor>, RecipientModifiedListener {
|
|
private final static String TAG = MessageDetailsActivity.class.getSimpleName();
|
|
|
|
public final static String MESSAGE_ID_EXTRA = "message_id";
|
|
public final static String THREAD_ID_EXTRA = "thread_id";
|
|
public final static String IS_PUSH_GROUP_EXTRA = "is_push_group";
|
|
public final static String TYPE_EXTRA = "type";
|
|
public final static String ADDRESS_EXTRA = "address";
|
|
|
|
private GlideRequests glideRequests;
|
|
private long threadId;
|
|
private boolean isPushGroup;
|
|
private ConversationItem conversationItem;
|
|
private ViewGroup itemParent;
|
|
private View metadataContainer;
|
|
private View expiresContainer;
|
|
private TextView errorText;
|
|
private View resendButton;
|
|
private TextView sentDate;
|
|
private TextView receivedDate;
|
|
private TextView expiresInText;
|
|
private View receivedContainer;
|
|
private TextView transport;
|
|
private TextView toFrom;
|
|
private ListView recipientsList;
|
|
private LayoutInflater inflater;
|
|
|
|
private DynamicTheme dynamicTheme = new DynamicTheme();
|
|
private DynamicLanguage dynamicLanguage = new DynamicLanguage();
|
|
|
|
private boolean running;
|
|
|
|
@Override
|
|
protected void onPreCreate() {
|
|
dynamicTheme.onCreate(this);
|
|
dynamicLanguage.onCreate(this);
|
|
}
|
|
|
|
@Override
|
|
public void onCreate(Bundle bundle, boolean ready) {
|
|
setContentView(R.layout.message_details_activity);
|
|
running = true;
|
|
|
|
initializeResources();
|
|
initializeActionBar();
|
|
getSupportLoaderManager().initLoader(0, null, this);
|
|
}
|
|
|
|
@Override
|
|
protected void onResume() {
|
|
super.onResume();
|
|
dynamicTheme.onResume(this);
|
|
dynamicLanguage.onResume(this);
|
|
|
|
assert getSupportActionBar() != null;
|
|
getSupportActionBar().setTitle(R.string.AndroidManifest__message_details);
|
|
|
|
MessageNotifier.setVisibleThread(threadId);
|
|
}
|
|
|
|
@Override
|
|
protected void onPause() {
|
|
super.onPause();
|
|
MessageNotifier.setVisibleThread(-1L);
|
|
}
|
|
|
|
@Override
|
|
protected void onDestroy() {
|
|
super.onDestroy();
|
|
running = false;
|
|
}
|
|
|
|
private void initializeActionBar() {
|
|
assert getSupportActionBar() != null;
|
|
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
|
|
|
Recipient recipient = Recipient.from(this, getIntent().getParcelableExtra(ADDRESS_EXTRA), true);
|
|
recipient.addListener(this);
|
|
|
|
setActionBarColor(recipient.getColor());
|
|
}
|
|
|
|
private void setActionBarColor(MaterialColor color) {
|
|
assert getSupportActionBar() != null;
|
|
getSupportActionBar().setBackgroundDrawable(new ColorDrawable(color.toActionBarColor(this)));
|
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
|
|
getWindow().setStatusBarColor(color.toStatusBarColor(this));
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onModified(final Recipient recipient) {
|
|
Util.runOnMain(() -> setActionBarColor(recipient.getColor()));
|
|
}
|
|
|
|
private void initializeResources() {
|
|
inflater = LayoutInflater.from(this);
|
|
View header = inflater.inflate(R.layout.message_details_header, recipientsList, false);
|
|
|
|
threadId = getIntent().getLongExtra(THREAD_ID_EXTRA, -1);
|
|
isPushGroup = getIntent().getBooleanExtra(IS_PUSH_GROUP_EXTRA, false);
|
|
glideRequests = GlideApp.with(this);
|
|
itemParent = header.findViewById(R.id.item_container);
|
|
recipientsList = findViewById(R.id.recipients_list);
|
|
metadataContainer = header.findViewById(R.id.metadata_container);
|
|
errorText = header.findViewById(R.id.error_text);
|
|
resendButton = header.findViewById(R.id.resend_button);
|
|
sentDate = header.findViewById(R.id.sent_time);
|
|
receivedContainer = header.findViewById(R.id.received_container);
|
|
receivedDate = header.findViewById(R.id.received_time);
|
|
transport = header.findViewById(R.id.transport);
|
|
toFrom = header.findViewById(R.id.tofrom);
|
|
expiresContainer = header.findViewById(R.id.expires_container);
|
|
expiresInText = header.findViewById(R.id.expires_in);
|
|
recipientsList.setHeaderDividersEnabled(false);
|
|
recipientsList.addHeaderView(header, null, false);
|
|
}
|
|
|
|
private void updateTransport(MessageRecord messageRecord) {
|
|
final String transportText;
|
|
if (messageRecord.isOutgoing() && messageRecord.isFailed()) {
|
|
transportText = "-";
|
|
} else if (messageRecord.isPending()) {
|
|
transportText = getString(R.string.ConversationFragment_pending);
|
|
} else if (messageRecord.isPush()) {
|
|
transportText = getString(R.string.ConversationFragment_push);
|
|
} else if (messageRecord.isMms()) {
|
|
transportText = getString(R.string.ConversationFragment_mms);
|
|
} else {
|
|
transportText = getString(R.string.ConversationFragment_sms);
|
|
}
|
|
|
|
transport.setText(transportText);
|
|
}
|
|
|
|
private void updateTime(MessageRecord messageRecord) {
|
|
sentDate.setOnLongClickListener(null);
|
|
receivedDate.setOnLongClickListener(null);
|
|
|
|
if (messageRecord.isPending() || messageRecord.isFailed()) {
|
|
sentDate.setText("-");
|
|
receivedContainer.setVisibility(View.GONE);
|
|
} else {
|
|
Locale dateLocale = dynamicLanguage.getCurrentLocale();
|
|
SimpleDateFormat dateFormatter = DateUtils.getDetailedDateFormatter(this, dateLocale);
|
|
sentDate.setText(dateFormatter.format(new Date(messageRecord.getDateSent())));
|
|
sentDate.setOnLongClickListener(v -> {
|
|
copyToClipboard(String.valueOf(messageRecord.getDateSent()));
|
|
return true;
|
|
});
|
|
|
|
if (messageRecord.getDateReceived() != messageRecord.getDateSent() && !messageRecord.isOutgoing()) {
|
|
receivedDate.setText(dateFormatter.format(new Date(messageRecord.getDateReceived())));
|
|
receivedDate.setOnLongClickListener(v -> {
|
|
copyToClipboard(String.valueOf(messageRecord.getDateReceived()));
|
|
return true;
|
|
});
|
|
receivedContainer.setVisibility(View.VISIBLE);
|
|
} else {
|
|
receivedContainer.setVisibility(View.GONE);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void updateExpirationTime(final MessageRecord messageRecord) {
|
|
if (messageRecord.getExpiresIn() <= 0 || messageRecord.getExpireStarted() <= 0) {
|
|
expiresContainer.setVisibility(View.GONE);
|
|
return;
|
|
}
|
|
|
|
expiresContainer.setVisibility(View.VISIBLE);
|
|
Util.runOnMain(new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
long elapsed = System.currentTimeMillis() - messageRecord.getExpireStarted();
|
|
long remaining = messageRecord.getExpiresIn() - elapsed;
|
|
|
|
String duration = ExpirationUtil.getExpirationDisplayValue(MessageDetailsActivity.this, Math.max((int)(remaining / 1000), 1));
|
|
expiresInText.setText(duration);
|
|
|
|
if (running) {
|
|
Util.runOnMainDelayed(this, 500);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
private void updateRecipients(MessageRecord messageRecord, Recipient recipient, List<RecipientDeliveryStatus> recipients) {
|
|
final int toFromRes;
|
|
if (messageRecord.isMms() && !messageRecord.isPush() && !messageRecord.isOutgoing()) {
|
|
toFromRes = R.string.message_details_header__with;
|
|
} else if (messageRecord.isOutgoing()) {
|
|
toFromRes = R.string.message_details_header__to;
|
|
} else {
|
|
toFromRes = R.string.message_details_header__from;
|
|
}
|
|
toFrom.setText(toFromRes);
|
|
conversationItem.bind(messageRecord, Optional.absent(), Optional.absent(), glideRequests, dynamicLanguage.getCurrentLocale(), new HashSet<>(), recipient, null, false);
|
|
recipientsList.setAdapter(new MessageDetailsRecipientAdapter(this, glideRequests, messageRecord, recipients, isPushGroup));
|
|
}
|
|
|
|
private void inflateMessageViewIfAbsent(MessageRecord messageRecord) {
|
|
if (conversationItem == null) {
|
|
if (messageRecord.isGroupAction()) {
|
|
conversationItem = (ConversationItem) inflater.inflate(R.layout.conversation_item_update, itemParent, false);
|
|
} else if (messageRecord.isOutgoing()) {
|
|
conversationItem = (ConversationItem) inflater.inflate(R.layout.conversation_item_sent, itemParent, false);
|
|
} else {
|
|
conversationItem = (ConversationItem) inflater.inflate(R.layout.conversation_item_received, itemParent, false);
|
|
}
|
|
itemParent.addView(conversationItem);
|
|
}
|
|
}
|
|
|
|
private @Nullable MessageRecord getMessageRecord(Context context, Cursor cursor, String type) {
|
|
switch (type) {
|
|
case MmsSmsDatabase.SMS_TRANSPORT:
|
|
SmsDatabase smsDatabase = DatabaseFactory.getSmsDatabase(context);
|
|
SmsDatabase.Reader reader = smsDatabase.readerFor(cursor);
|
|
return reader.getNext();
|
|
case MmsSmsDatabase.MMS_TRANSPORT:
|
|
MmsDatabase mmsDatabase = DatabaseFactory.getMmsDatabase(context);
|
|
MmsDatabase.Reader mmsReader = mmsDatabase.readerFor(cursor);
|
|
return mmsReader.getNext();
|
|
default:
|
|
throw new AssertionError("no valid message type specified");
|
|
}
|
|
}
|
|
|
|
private void copyToClipboard(@NonNull String text) {
|
|
((ClipboardManager) getSystemService(Context.CLIPBOARD_SERVICE)).setPrimaryClip(ClipData.newPlainText("text", text));
|
|
}
|
|
|
|
@Override
|
|
public @NonNull Loader<Cursor> onCreateLoader(int id, Bundle args) {
|
|
return new MessageDetailsLoader(this, getIntent().getStringExtra(TYPE_EXTRA),
|
|
getIntent().getLongExtra(MESSAGE_ID_EXTRA, -1));
|
|
}
|
|
|
|
@Override
|
|
public void onLoadFinished(@NonNull Loader<Cursor> loader, Cursor cursor) {
|
|
MessageRecord messageRecord = getMessageRecord(this, cursor, getIntent().getStringExtra(TYPE_EXTRA));
|
|
|
|
if (messageRecord == null) {
|
|
finish();
|
|
} else {
|
|
new MessageRecipientAsyncTask(this, messageRecord).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onLoaderReset(@NonNull Loader<Cursor> loader) {
|
|
recipientsList.setAdapter(null);
|
|
}
|
|
|
|
@Override
|
|
public boolean onOptionsItemSelected(MenuItem item) {
|
|
super.onOptionsItemSelected(item);
|
|
|
|
switch (item.getItemId()) {
|
|
case android.R.id.home: finish(); return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
@SuppressLint("StaticFieldLeak")
|
|
private class MessageRecipientAsyncTask extends AsyncTask<Void,Void,List<RecipientDeliveryStatus>> {
|
|
|
|
private final WeakReference<Context> weakContext;
|
|
private final MessageRecord messageRecord;
|
|
|
|
MessageRecipientAsyncTask(@NonNull Context context, @NonNull MessageRecord messageRecord) {
|
|
this.weakContext = new WeakReference<>(context);
|
|
this.messageRecord = messageRecord;
|
|
}
|
|
|
|
protected Context getContext() {
|
|
return weakContext.get();
|
|
}
|
|
|
|
@Override
|
|
public List<RecipientDeliveryStatus> doInBackground(Void... voids) {
|
|
Context context = getContext();
|
|
|
|
if (context == null) {
|
|
Log.w(TAG, "associated context is destroyed, finishing early");
|
|
return null;
|
|
}
|
|
|
|
List<RecipientDeliveryStatus> recipients = new LinkedList<>();
|
|
|
|
if (!messageRecord.getRecipient().isGroupRecipient()) {
|
|
recipients.add(new RecipientDeliveryStatus(messageRecord.getRecipient(), getStatusFor(messageRecord.getDeliveryReceiptCount(), messageRecord.getReadReceiptCount(), messageRecord.isPending()), messageRecord.isUnidentified(), -1));
|
|
} else {
|
|
List<GroupReceiptInfo> receiptInfoList = DatabaseFactory.getGroupReceiptDatabase(context).getGroupReceiptInfo(messageRecord.getId());
|
|
|
|
if (receiptInfoList.isEmpty()) {
|
|
List<Recipient> group = DatabaseFactory.getGroupDatabase(context).getGroupMembers(messageRecord.getRecipient().getAddress().toGroupString(), false);
|
|
|
|
for (Recipient recipient : group) {
|
|
recipients.add(new RecipientDeliveryStatus(recipient, RecipientDeliveryStatus.Status.UNKNOWN, false, -1));
|
|
}
|
|
} else {
|
|
for (GroupReceiptInfo info : receiptInfoList) {
|
|
recipients.add(new RecipientDeliveryStatus(Recipient.from(context, info.getAddress(), true),
|
|
getStatusFor(info.getStatus(), messageRecord.isPending(), messageRecord.isFailed()),
|
|
info.isUnidentified(),
|
|
info.getTimestamp()));
|
|
}
|
|
}
|
|
}
|
|
|
|
return recipients;
|
|
}
|
|
|
|
@Override
|
|
public void onPostExecute(List<RecipientDeliveryStatus> recipients) {
|
|
if (getContext() == null) {
|
|
Log.w(TAG, "AsyncTask finished with a destroyed context, leaving early.");
|
|
return;
|
|
}
|
|
|
|
inflateMessageViewIfAbsent(messageRecord);
|
|
updateRecipients(messageRecord, messageRecord.getRecipient(), recipients);
|
|
|
|
boolean isGroupNetworkFailure = messageRecord.isFailed() && !messageRecord.getNetworkFailures().isEmpty();
|
|
boolean isIndividualNetworkFailure = messageRecord.isFailed() && !isPushGroup && messageRecord.getIdentityKeyMismatches().isEmpty();
|
|
|
|
if (isGroupNetworkFailure || isIndividualNetworkFailure) {
|
|
errorText.setVisibility(View.VISIBLE);
|
|
resendButton.setVisibility(View.VISIBLE);
|
|
resendButton.setOnClickListener(this::onResendClicked);
|
|
metadataContainer.setVisibility(View.GONE);
|
|
} else if (messageRecord.isFailed()) {
|
|
errorText.setVisibility(View.VISIBLE);
|
|
resendButton.setVisibility(View.GONE);
|
|
resendButton.setOnClickListener(null);
|
|
metadataContainer.setVisibility(View.GONE);
|
|
} else {
|
|
updateTransport(messageRecord);
|
|
updateTime(messageRecord);
|
|
updateExpirationTime(messageRecord);
|
|
errorText.setVisibility(View.GONE);
|
|
resendButton.setVisibility(View.GONE);
|
|
resendButton.setOnClickListener(null);
|
|
metadataContainer.setVisibility(View.VISIBLE);
|
|
}
|
|
}
|
|
|
|
private RecipientDeliveryStatus.Status getStatusFor(int deliveryReceiptCount, int readReceiptCount, boolean pending) {
|
|
if (readReceiptCount > 0) return RecipientDeliveryStatus.Status.READ;
|
|
else if (deliveryReceiptCount > 0) return RecipientDeliveryStatus.Status.DELIVERED;
|
|
else if (!pending) return RecipientDeliveryStatus.Status.SENT;
|
|
else return RecipientDeliveryStatus.Status.PENDING;
|
|
}
|
|
|
|
private RecipientDeliveryStatus.Status getStatusFor(int groupStatus, boolean pending, boolean failed) {
|
|
if (groupStatus == GroupReceiptDatabase.STATUS_READ) return RecipientDeliveryStatus.Status.READ;
|
|
else if (groupStatus == GroupReceiptDatabase.STATUS_DELIVERED) return RecipientDeliveryStatus.Status.DELIVERED;
|
|
else if (groupStatus == GroupReceiptDatabase.STATUS_UNDELIVERED && failed) return RecipientDeliveryStatus.Status.UNKNOWN;
|
|
else if (groupStatus == GroupReceiptDatabase.STATUS_UNDELIVERED && !pending) return RecipientDeliveryStatus.Status.SENT;
|
|
else if (groupStatus == GroupReceiptDatabase.STATUS_UNDELIVERED) return RecipientDeliveryStatus.Status.PENDING;
|
|
else if (groupStatus == GroupReceiptDatabase.STATUS_UNKNOWN) return RecipientDeliveryStatus.Status.UNKNOWN;
|
|
throw new AssertionError();
|
|
}
|
|
|
|
private void onResendClicked(View v) {
|
|
MessageSender.resend(MessageDetailsActivity.this, messageRecord);
|
|
resendButton.setVisibility(View.GONE);
|
|
}
|
|
}
|
|
}
|