/* * 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 . */ package org.thoughtcrime.securesms.conversation; import android.annotation.SuppressLint; import android.app.Activity; import android.content.ClipData; import android.content.ClipboardManager; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.net.Uri; import android.os.AsyncTask; import android.os.Build; import android.os.Bundle; import android.text.SpannableStringBuilder; import android.text.TextUtils; import android.view.LayoutInflater; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; import android.view.Window; import android.view.animation.Animation; import android.view.animation.AnimationUtils; import android.widget.FrameLayout; import android.widget.TextView; import android.widget.Toast; import android.widget.ViewSwitcher; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.appcompat.app.AlertDialog; import androidx.appcompat.app.AppCompatActivity; import androidx.appcompat.view.ActionMode; import androidx.appcompat.widget.Toolbar; import androidx.core.app.ActivityCompat; import androidx.core.app.ActivityOptionsCompat; import androidx.core.text.HtmlCompat; import androidx.lifecycle.ViewModelProviders; import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; import androidx.recyclerview.widget.RecyclerView.OnScrollListener; import com.annimon.stream.Collectors; import com.annimon.stream.Stream; import com.google.android.collect.Sets; import org.thoughtcrime.securesms.ApplicationContext; import org.thoughtcrime.securesms.LoggingFragment; import org.thoughtcrime.securesms.PassphraseRequiredActivity; import org.thoughtcrime.securesms.R; import org.thoughtcrime.securesms.attachments.Attachment; import org.thoughtcrime.securesms.components.ConversationScrollToView; import org.thoughtcrime.securesms.components.ConversationTypingView; import org.thoughtcrime.securesms.components.TooltipPopup; import org.thoughtcrime.securesms.components.recyclerview.SmoothScrollingLinearLayoutManager; import org.thoughtcrime.securesms.contactshare.Contact; import org.thoughtcrime.securesms.contactshare.ContactUtil; import org.thoughtcrime.securesms.contactshare.SharedContactDetailsActivity; import org.thoughtcrime.securesms.conversation.ConversationAdapter.ItemClickListener; import org.thoughtcrime.securesms.conversation.ConversationAdapter.StickyHeaderViewHolder; import org.thoughtcrime.securesms.conversation.ConversationMessage.ConversationMessageFactory; import org.thoughtcrime.securesms.database.DatabaseFactory; import org.thoughtcrime.securesms.database.MessageDatabase; import org.thoughtcrime.securesms.database.MmsDatabase; import org.thoughtcrime.securesms.database.RecipientDatabase; import org.thoughtcrime.securesms.database.SmsDatabase; import org.thoughtcrime.securesms.database.model.MediaMmsMessageRecord; import org.thoughtcrime.securesms.database.model.MessageRecord; import org.thoughtcrime.securesms.database.model.MmsMessageRecord; import org.thoughtcrime.securesms.dependencies.ApplicationDependencies; import org.thoughtcrime.securesms.groups.GroupId; import org.thoughtcrime.securesms.jobs.DirectoryRefreshJob; import org.thoughtcrime.securesms.jobs.MultiDeviceViewOnceOpenJob; import org.thoughtcrime.securesms.keyvalue.SignalStore; import org.thoughtcrime.securesms.linkpreview.LinkPreview; import org.thoughtcrime.securesms.logging.Log; import org.thoughtcrime.securesms.longmessage.LongMessageActivity; import org.thoughtcrime.securesms.mediasend.Media; import org.thoughtcrime.securesms.messagedetails.MessageDetailsActivity; import org.thoughtcrime.securesms.messagerequests.MessageRequestViewModel; import org.thoughtcrime.securesms.mms.GlideApp; import org.thoughtcrime.securesms.mms.OutgoingMediaMessage; import org.thoughtcrime.securesms.mms.PartAuthority; import org.thoughtcrime.securesms.mms.Slide; import org.thoughtcrime.securesms.profiles.UnknownSenderView; import org.thoughtcrime.securesms.providers.BlobProvider; import org.thoughtcrime.securesms.reactions.ReactionsBottomSheetDialogFragment; import org.thoughtcrime.securesms.recipients.LiveRecipient; import org.thoughtcrime.securesms.recipients.Recipient; import org.thoughtcrime.securesms.recipients.RecipientId; import org.thoughtcrime.securesms.recipients.ui.bottomsheet.RecipientBottomSheetDialogFragment; import org.thoughtcrime.securesms.revealable.ViewOnceMessageActivity; import org.thoughtcrime.securesms.revealable.ViewOnceUtil; import org.thoughtcrime.securesms.sharing.ShareActivity; import org.thoughtcrime.securesms.sms.MessageSender; import org.thoughtcrime.securesms.sms.OutgoingTextMessage; import org.thoughtcrime.securesms.stickers.StickerLocator; import org.thoughtcrime.securesms.stickers.StickerPackPreviewActivity; import org.thoughtcrime.securesms.util.CachedInflater; import org.thoughtcrime.securesms.util.CommunicationActions; import org.thoughtcrime.securesms.util.FeatureFlags; import org.thoughtcrime.securesms.util.HtmlUtil; import org.thoughtcrime.securesms.util.RemoteDeleteUtil; import org.thoughtcrime.securesms.util.SaveAttachmentTask; import org.thoughtcrime.securesms.util.SnapToTopDataObserver; import org.thoughtcrime.securesms.util.StickyHeaderDecoration; import org.thoughtcrime.securesms.util.TextSecurePreferences; import org.thoughtcrime.securesms.util.Util; import org.thoughtcrime.securesms.util.ViewUtil; import org.thoughtcrime.securesms.util.concurrent.SignalExecutors; import org.thoughtcrime.securesms.util.concurrent.SimpleTask; import org.thoughtcrime.securesms.util.task.ProgressDialogAsyncTask; import org.thoughtcrime.securesms.util.views.AdaptiveActionsToolbar; import org.whispersystems.libsignal.util.guava.Optional; import java.io.IOException; import java.io.InputStream; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Locale; import java.util.Set; @SuppressLint("StaticFieldLeak") public class ConversationFragment extends LoggingFragment { private static final String TAG = ConversationFragment.class.getSimpleName(); private static final int SCROLL_ANIMATION_THRESHOLD = 50; private static final int CODE_ADD_EDIT_CONTACT = 77; private final ActionModeCallback actionModeCallback = new ActionModeCallback(); private final ItemClickListener selectionClickListener = new ConversationFragmentItemClickListener(); private ConversationFragmentListener listener; private LiveRecipient recipient; private long threadId; private boolean isReacting; private ActionMode actionMode; private Locale locale; private RecyclerView list; private RecyclerView.ItemDecoration lastSeenDecoration; private RecyclerView.ItemDecoration stickyHeaderDecoration; private ViewSwitcher topLoadMoreView; private ViewSwitcher bottomLoadMoreView; private ConversationTypingView typingView; private UnknownSenderView unknownSenderView; private View composeDivider; private ConversationScrollToView scrollToBottomButton; private ConversationScrollToView scrollToMentionButton; private TextView scrollDateHeader; private ConversationBannerView conversationBanner; private ConversationBannerView emptyConversationBanner; private MessageRequestViewModel messageRequestViewModel; private MessageCountsViewModel messageCountsViewModel; private ConversationViewModel conversationViewModel; private SnapToTopDataObserver snapToTopDataObserver; private MarkReadHelper markReadHelper; private Animation scrollButtonInAnimation; private Animation mentionButtonInAnimation; private Animation scrollButtonOutAnimation; private Animation mentionButtonOutAnimation; private OnScrollListener conversationScrollListener; private int pulsePosition = -1; public static void prepare(@NonNull Context context) { FrameLayout parent = new FrameLayout(context); parent.setLayoutParams(new FrameLayout.LayoutParams(FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.WRAP_CONTENT)); CachedInflater.from(context).cacheUntilLimit(R.layout.conversation_item_received_text_only, parent, 15); CachedInflater.from(context).cacheUntilLimit(R.layout.conversation_item_sent_text_only, parent, 15); CachedInflater.from(context).cacheUntilLimit(R.layout.conversation_item_received_multimedia, parent, 10); CachedInflater.from(context).cacheUntilLimit(R.layout.conversation_item_sent_multimedia, parent, 10); CachedInflater.from(context).cacheUntilLimit(R.layout.conversation_item_update, parent, 5); CachedInflater.from(context).cacheUntilLimit(R.layout.cursor_adapter_header_footer_view, parent, 2); } @Override public void onCreate(Bundle icicle) { super.onCreate(icicle); this.locale = (Locale) getArguments().getSerializable(PassphraseRequiredActivity.LOCALE_EXTRA); } @Override public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, Bundle bundle) { final View view = inflater.inflate(R.layout.conversation_fragment, container, false); list = view.findViewById(android.R.id.list); composeDivider = view.findViewById(R.id.compose_divider); scrollToBottomButton = view.findViewById(R.id.scroll_to_bottom); scrollToMentionButton = view.findViewById(R.id.scroll_to_mention); scrollDateHeader = view.findViewById(R.id.scroll_date_header); emptyConversationBanner = view.findViewById(R.id.empty_conversation_banner); final LinearLayoutManager layoutManager = new SmoothScrollingLinearLayoutManager(getActivity(), true); list.setHasFixedSize(false); list.setLayoutManager(layoutManager); list.setItemAnimator(null); snapToTopDataObserver = new ConversationSnapToTopDataObserver(list, new ConversationScrollRequestValidator()); conversationBanner = (ConversationBannerView) inflater.inflate(R.layout.conversation_item_banner, container, false); topLoadMoreView = (ViewSwitcher) inflater.inflate(R.layout.load_more_header, container, false); bottomLoadMoreView = (ViewSwitcher) inflater.inflate(R.layout.load_more_header, container, false); initializeLoadMoreView(topLoadMoreView); initializeLoadMoreView(bottomLoadMoreView); typingView = (ConversationTypingView) inflater.inflate(R.layout.conversation_typing_view, container, false); new ConversationItemSwipeCallback( conversationMessage -> actionMode == null && MenuState.canReplyToMessage(recipient.get(), MenuState.isActionMessage(conversationMessage.getMessageRecord()), conversationMessage.getMessageRecord(), messageRequestViewModel.shouldShowMessageRequest()), this::handleReplyMessage ).attachToRecyclerView(list); setupListLayoutListeners(); this.messageCountsViewModel = ViewModelProviders.of(requireActivity()).get(MessageCountsViewModel.class); this.conversationViewModel = ViewModelProviders.of(requireActivity(), new ConversationViewModel.Factory()).get(ConversationViewModel.class); conversationViewModel.getMessages().observe(this, list -> { if (getListAdapter() != null && !list.getDataSource().isInvalid()) { Log.i(TAG, "submitList"); getListAdapter().submitList(list); } else if (list.getDataSource().isInvalid()) { Log.i(TAG, "submitList skipped an invalid list"); } }); conversationViewModel.getConversationMetadata().observe(this, this::presentConversationMetadata); conversationViewModel.getShowMentionsButton().observe(this, shouldShow -> { if (shouldShow) { ViewUtil.animateIn(scrollToMentionButton, mentionButtonInAnimation); } else { ViewUtil.animateOut(scrollToMentionButton, mentionButtonOutAnimation, View.INVISIBLE); } }); conversationViewModel.getShowScrollToBottom().observe(this, shouldShow -> { if (shouldShow) { ViewUtil.animateIn(scrollToBottomButton, scrollButtonInAnimation); } else { ViewUtil.animateOut(scrollToBottomButton, scrollButtonOutAnimation, View.INVISIBLE); } }); scrollToBottomButton.setOnClickListener(v -> scrollToBottom()); scrollToMentionButton.setOnClickListener(v -> scrollToNextMention()); return view; } private void setupListLayoutListeners() { list.addOnLayoutChangeListener((v, left, top, right, bottom, oldLeft, oldTop, oldRight, oldBottom) -> setListVerticalTranslation()); list.addOnChildAttachStateChangeListener(new RecyclerView.OnChildAttachStateChangeListener() { @Override public void onChildViewAttachedToWindow(@NonNull View view) { setListVerticalTranslation(); } @Override public void onChildViewDetachedFromWindow(@NonNull View view) { setListVerticalTranslation(); } }); } private void setListVerticalTranslation() { if (list.canScrollVertically(1) || list.canScrollVertically(-1) || list.getChildCount() == 0) { list.setTranslationY(0); list.setOverScrollMode(RecyclerView.OVER_SCROLL_IF_CONTENT_SCROLLS); } else { int chTop = list.getChildAt(list.getChildCount() - 1).getTop(); list.setTranslationY(Math.min(0, -chTop)); list.setOverScrollMode(RecyclerView.OVER_SCROLL_NEVER); } listener.onListVerticalTranslationChanged(list.getTranslationY()); } @Override public void onActivityCreated(Bundle bundle) { super.onActivityCreated(bundle); initializeScrollButtonAnimations(); initializeResources(); initializeMessageRequestViewModel(); initializeListAdapter(); } @Override public void onAttach(Activity activity) { super.onAttach(activity); this.listener = (ConversationFragmentListener)activity; } @Override public void onStart() { super.onStart(); initializeTypingObserver(); } @Override public void onPause() { super.onPause(); int lastVisiblePosition = getListLayoutManager().findLastVisibleItemPosition(); int firstVisiblePosition = getListLayoutManager().findFirstCompletelyVisibleItemPosition(); final long lastVisibleMessageTimestamp; if (firstVisiblePosition > 0 && lastVisiblePosition != RecyclerView.NO_POSITION) { ConversationMessage message = getListAdapter().getLastVisibleConversationMessage(lastVisiblePosition); lastVisibleMessageTimestamp = message != null ? message.getMessageRecord().getDateReceived() : 0; } else { lastVisibleMessageTimestamp = 0; } SignalExecutors.BOUNDED.submit(() -> DatabaseFactory.getThreadDatabase(requireContext()).setLastScrolled(threadId, lastVisibleMessageTimestamp)); } @Override public void onStop() { super.onStop(); ApplicationContext.getInstance(requireContext()).getTypingStatusRepository().getTypists(threadId).removeObservers(this); } public void onNewIntent() { if (actionMode != null) { actionMode.finish(); } initializeResources(); messageRequestViewModel.setConversationInfo(recipient.getId(), threadId); initializeListAdapter(); } public void moveToLastSeen() { if (conversationViewModel.getLastSeenPosition() <= 0) { Log.i(TAG, "No need to move to last seen."); return; } if (list == null || getListAdapter() == null) { Log.w(TAG, "Tried to move to last seen position, but we hadn't initialized the view yet."); return; } int position = getListAdapter().getAdapterPositionForMessagePosition(conversationViewModel.getLastSeenPosition()); snapToTopDataObserver.requestScrollPosition(position); } private void initializeMessageRequestViewModel() { MessageRequestViewModel.Factory factory = new MessageRequestViewModel.Factory(requireContext()); messageRequestViewModel = ViewModelProviders.of(requireActivity(), factory).get(MessageRequestViewModel.class); messageRequestViewModel.setConversationInfo(recipient.getId(), threadId); listener.onMessageRequest(messageRequestViewModel); messageRequestViewModel.getRecipientInfo().observe(getViewLifecycleOwner(), recipientInfo -> { presentMessageRequestProfileView(requireContext(), recipientInfo, conversationBanner); presentMessageRequestProfileView(requireContext(), recipientInfo, emptyConversationBanner); }); } private static void presentMessageRequestProfileView(@NonNull Context context, @NonNull MessageRequestViewModel.RecipientInfo recipientInfo, @Nullable ConversationBannerView conversationBanner) { if (conversationBanner == null) { return; } Recipient recipient = recipientInfo.getRecipient(); boolean isSelf = Recipient.self().equals(recipient); int memberCount = recipientInfo.getGroupMemberCount(); int pendingMemberCount = recipientInfo.getGroupPendingMemberCount(); List groups = recipientInfo.getSharedGroups(); if (recipient != null) { conversationBanner.setAvatar(GlideApp.with(context), recipient); String title = isSelf ? context.getString(R.string.note_to_self) : recipient.getDisplayNameOrUsername(context); conversationBanner.setTitle(title); if (recipient.isGroup()) { if (pendingMemberCount > 0) { conversationBanner.setSubtitle(context.getResources() .getQuantityString(R.plurals.MessageRequestProfileView_members_and_invited, memberCount, memberCount, pendingMemberCount)); } else if (memberCount > 0) { conversationBanner.setSubtitle(context.getResources().getQuantityString(R.plurals.MessageRequestProfileView_members, memberCount, memberCount)); } else { conversationBanner.setSubtitle(null); } } else if (isSelf) { conversationBanner.setSubtitle(context.getString(R.string.ConversationFragment__you_can_add_notes_for_yourself_in_this_conversation)); } else { String subtitle = recipient.getE164().orNull(); if (subtitle == null || subtitle.equals(title)) { conversationBanner.hideSubtitle(); } else { conversationBanner.setSubtitle(subtitle); } } } if (groups.isEmpty() || isSelf) { conversationBanner.hideDescription(); } else { final String description; switch (groups.size()) { case 1: description = context.getString(R.string.MessageRequestProfileView_member_of_one_group, HtmlUtil.bold(groups.get(0))); break; case 2: description = context.getString(R.string.MessageRequestProfileView_member_of_two_groups, HtmlUtil.bold(groups.get(0)), HtmlUtil.bold(groups.get(1))); break; case 3: description = context.getString(R.string.MessageRequestProfileView_member_of_many_groups, HtmlUtil.bold(groups.get(0)), HtmlUtil.bold(groups.get(1)), HtmlUtil.bold(groups.get(2))); break; default: int others = groups.size() - 2; description = context.getString(R.string.MessageRequestProfileView_member_of_many_groups, HtmlUtil.bold(groups.get(0)), HtmlUtil.bold(groups.get(1)), context.getResources().getQuantityString(R.plurals.MessageRequestProfileView_member_of_d_additional_groups, others, others)); } conversationBanner.setDescription(HtmlCompat.fromHtml(description, 0)); conversationBanner.showDescription(); } } private void initializeResources() { long oldThreadId = threadId; int startingPosition = this.getActivity().getIntent().getIntExtra(ConversationActivity.STARTING_POSITION_EXTRA, -1); this.recipient = Recipient.live(getActivity().getIntent().getParcelableExtra(ConversationActivity.RECIPIENT_EXTRA)); this.threadId = this.getActivity().getIntent().getLongExtra(ConversationActivity.THREAD_ID_EXTRA, -1); this.unknownSenderView = new UnknownSenderView(getActivity(), recipient.get(), threadId, () -> clearHeaderIfNotTyping(getListAdapter())); this.markReadHelper = new MarkReadHelper(threadId, requireContext()); conversationViewModel.onConversationDataAvailable(threadId, startingPosition); messageCountsViewModel.setThreadId(threadId); messageCountsViewModel.getUnreadMessagesCount().observe(getViewLifecycleOwner(), scrollToBottomButton::setUnreadCount); messageCountsViewModel.getUnreadMentionsCount().observe(getViewLifecycleOwner(), count -> { scrollToMentionButton.setUnreadCount(count); conversationViewModel.setHasUnreadMentions(count > 0); }); conversationScrollListener = new ConversationScrollListener(requireContext()); list.addOnScrollListener(conversationScrollListener); if (oldThreadId != threadId) { ApplicationContext.getInstance(requireContext()).getTypingStatusRepository().getTypists(oldThreadId).removeObservers(this); } } private void initializeListAdapter() { if (this.recipient != null && this.threadId != -1) { Log.d(TAG, "Initializing adapter for " + recipient.getId()); ConversationAdapter adapter = new ConversationAdapter(this, GlideApp.with(this), locale, selectionClickListener, this.recipient.get()); list.setAdapter(adapter); setStickyHeaderDecoration(adapter); ConversationAdapter.initializePool(list.getRecycledViewPool()); adapter.registerAdapterDataObserver(snapToTopDataObserver); setLastSeen(conversationViewModel.getLastSeen()); emptyConversationBanner.setVisibility(View.GONE); } else if (threadId == -1) { emptyConversationBanner.setVisibility(View.VISIBLE); } } private void initializeLoadMoreView(ViewSwitcher loadMoreView) { loadMoreView.setOnClickListener(v -> { loadMoreView.showNext(); loadMoreView.setOnClickListener(null); }); } private void initializeTypingObserver() { if (!TextSecurePreferences.isTypingIndicatorsEnabled(requireContext())) { return; } ApplicationContext.getInstance(requireContext()).getTypingStatusRepository().getTypists(threadId).removeObservers(this); ApplicationContext.getInstance(requireContext()).getTypingStatusRepository().getTypists(threadId).observe(this, typingState -> { List recipients; boolean replacedByIncomingMessage; if (typingState != null) { recipients = typingState.getTypists(); replacedByIncomingMessage = typingState.isReplacedByIncomingMessage(); } else { recipients = Collections.emptyList(); replacedByIncomingMessage = false; } typingView.setTypists(GlideApp.with(ConversationFragment.this), recipients, recipient.get().isGroup()); ConversationAdapter adapter = getListAdapter(); if (adapter.getHeaderView() != null && adapter.getHeaderView() != typingView) { Log.i(TAG, "Skipping typing indicator -- the header slot is occupied."); return; } if (recipients.size() > 0) { if (!isTypingIndicatorShowing() && isAtBottom()) { Context context = requireContext(); list.setVerticalScrollBarEnabled(false); list.post(() -> { if (!isReacting) { getListLayoutManager().smoothScrollToPosition(context, 0, 250); } }); list.postDelayed(() -> list.setVerticalScrollBarEnabled(true), 300); adapter.setHeaderView(typingView); } else { if (isTypingIndicatorShowing()) { adapter.setHeaderView(typingView); } else { adapter.setHeaderView(typingView); } } } else { if (isTypingIndicatorShowing() && getListLayoutManager().findFirstCompletelyVisibleItemPosition() == 0 && getListLayoutManager().getItemCount() > 1 && !replacedByIncomingMessage) { if (!isReacting) { getListLayoutManager().smoothScrollToPosition(requireContext(), 1, 250); } list.setVerticalScrollBarEnabled(false); list.postDelayed(() -> { adapter.setHeaderView(null); list.post(() -> list.setVerticalScrollBarEnabled(true)); }, 200); } else if (!replacedByIncomingMessage) { adapter.setHeaderView(null); } else { adapter.setHeaderView(null); } } }); } private void setCorrectMenuVisibility(@NonNull Menu menu) { Set messages = getListAdapter().getSelectedItems(); if (actionMode != null && messages.size() == 0) { actionMode.finish(); return; } MenuState menuState = MenuState.getMenuState(recipient.get(), Stream.of(messages).map(ConversationMessage::getMessageRecord).collect(Collectors.toSet()), messageRequestViewModel.shouldShowMessageRequest()); menu.findItem(R.id.menu_context_forward).setVisible(menuState.shouldShowForwardAction()); menu.findItem(R.id.menu_context_reply).setVisible(menuState.shouldShowReplyAction()); menu.findItem(R.id.menu_context_details).setVisible(menuState.shouldShowDetailsAction()); menu.findItem(R.id.menu_context_save_attachment).setVisible(menuState.shouldShowSaveAttachmentAction()); menu.findItem(R.id.menu_context_resend).setVisible(menuState.shouldShowResendAction()); menu.findItem(R.id.menu_context_copy).setVisible(menuState.shouldShowCopyAction()); } private ConversationAdapter getListAdapter() { return (ConversationAdapter) list.getAdapter(); } private SmoothScrollingLinearLayoutManager getListLayoutManager() { return (SmoothScrollingLinearLayoutManager) list.getLayoutManager(); } private ConversationMessage getSelectedConversationMessage() { Set messageRecords = getListAdapter().getSelectedItems(); if (messageRecords.size() == 1) return messageRecords.iterator().next(); else throw new AssertionError(); } public void reload(Recipient recipient, long threadId) { this.recipient = recipient.live(); if (this.threadId != threadId) { this.threadId = threadId; messageRequestViewModel.setConversationInfo(recipient.getId(), threadId); snapToTopDataObserver.requestScrollPosition(0); conversationViewModel.onConversationDataAvailable(threadId, -1); messageCountsViewModel.setThreadId(threadId); initializeListAdapter(); } } public void scrollToBottom() { if (getListLayoutManager().findFirstVisibleItemPosition() < SCROLL_ANIMATION_THRESHOLD) { list.smoothScrollToPosition(0); } else { list.scrollToPosition(0); } } public void setStickyHeaderDecoration(@NonNull ConversationAdapter adapter) { if (stickyHeaderDecoration != null) { list.removeItemDecoration(stickyHeaderDecoration); } stickyHeaderDecoration = new StickyHeaderDecoration(adapter, false, false); list.addItemDecoration(stickyHeaderDecoration); } public void setLastSeen(long lastSeen) { if (lastSeenDecoration != null) { list.removeItemDecoration(lastSeenDecoration); } lastSeenDecoration = new LastSeenHeader(getListAdapter(), lastSeen); list.addItemDecoration(lastSeenDecoration); } private void handleCopyMessage(final Set conversationMessages) { List messageList = new ArrayList<>(conversationMessages); Collections.sort(messageList, (lhs, rhs) -> Long.compare(lhs.getMessageRecord().getDateReceived(), rhs.getMessageRecord().getDateReceived())); SpannableStringBuilder bodyBuilder = new SpannableStringBuilder(); ClipboardManager clipboard = (ClipboardManager) requireActivity().getSystemService(Context.CLIPBOARD_SERVICE); for (ConversationMessage message : messageList) { CharSequence body = message.getDisplayBody(requireContext()); if (!TextUtils.isEmpty(body)) { if (bodyBuilder.length() > 0) { bodyBuilder.append('\n'); } bodyBuilder.append(body); } } if (!TextUtils.isEmpty(bodyBuilder)) { clipboard.setPrimaryClip(ClipData.newPlainText(null, bodyBuilder)); } } private void handleDeleteMessages(final Set conversationMessages) { Set messageRecords = Stream.of(conversationMessages).map(ConversationMessage::getMessageRecord).collect(Collectors.toSet()); buildRemoteDeleteConfirmationDialog(messageRecords).show(); } private AlertDialog.Builder buildRemoteDeleteConfirmationDialog(Set messageRecords) { Context context = requireActivity(); int messagesCount = messageRecords.size(); AlertDialog.Builder builder = new AlertDialog.Builder(getActivity()); builder.setTitle(getActivity().getResources().getQuantityString(R.plurals.ConversationFragment_delete_selected_messages, messagesCount, messagesCount)); builder.setCancelable(true); builder.setPositiveButton(R.string.ConversationFragment_delete_for_me, (dialog, which) -> { new ProgressDialogAsyncTask(getActivity(), R.string.ConversationFragment_deleting, R.string.ConversationFragment_deleting_messages) { @Override protected Void doInBackground(Void... voids) { for (MessageRecord messageRecord : messageRecords) { boolean threadDeleted; if (messageRecord.isMms()) { threadDeleted = DatabaseFactory.getMmsDatabase(context).deleteMessage(messageRecord.getId()); } else { threadDeleted = DatabaseFactory.getSmsDatabase(context).deleteMessage(messageRecord.getId()); } if (threadDeleted) { threadId = -1; conversationViewModel.clearThreadId(); messageCountsViewModel.clearThreadId(); listener.setThreadId(threadId); } } return null; } }.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); }); if (RemoteDeleteUtil.isValidSend(messageRecords, System.currentTimeMillis())) { builder.setNeutralButton(R.string.ConversationFragment_delete_for_everyone, (dialog, which) -> handleDeleteForEveryone(messageRecords)); } builder.setNegativeButton(android.R.string.cancel, null); return builder; } private void handleDeleteForEveryone(Set messageRecords) { Runnable deleteForEveryone = () -> { SignalExecutors.BOUNDED.execute(() -> { for (MessageRecord message : messageRecords) { MessageSender.sendRemoteDelete(ApplicationDependencies.getApplication(), message.getId(), message.isMms()); } }); }; if (SignalStore.uiHints().hasConfirmedDeleteForEveryoneOnce()) { deleteForEveryone.run(); } else { new AlertDialog.Builder(requireActivity()) .setMessage(R.string.ConversationFragment_this_message_will_be_deleted_for_everyone_in_the_conversation) .setPositiveButton(R.string.ConversationFragment_delete_for_everyone, (dialog, which) -> { SignalStore.uiHints().markHasConfirmedDeleteForEveryoneOnce(); deleteForEveryone.run(); }) .setNegativeButton(android.R.string.cancel, null) .show(); } } private void handleDisplayDetails(ConversationMessage message) { startActivity(MessageDetailsActivity.getIntentForMessageDetails(requireContext(), message.getMessageRecord(), recipient.getId(), threadId)); } private void handleForwardMessage(ConversationMessage conversationMessage) { if (conversationMessage.getMessageRecord().isViewOnce()) { throw new AssertionError("Cannot forward a view-once message."); } listener.onForwardClicked(); SimpleTask.run(getLifecycle(), () -> { Intent composeIntent = new Intent(getActivity(), ShareActivity.class); composeIntent.putExtra(Intent.EXTRA_TEXT, conversationMessage.getDisplayBody(requireContext())); if (conversationMessage.getMessageRecord().isMms()) { MmsMessageRecord mediaMessage = (MmsMessageRecord) conversationMessage.getMessageRecord(); boolean isAlbum = mediaMessage.containsMediaSlide() && mediaMessage.getSlideDeck().getSlides().size() > 1 && mediaMessage.getSlideDeck().getAudioSlide() == null && mediaMessage.getSlideDeck().getDocumentSlide() == null && mediaMessage.getSlideDeck().getStickerSlide() == null; if (isAlbum) { ArrayList mediaList = new ArrayList<>(mediaMessage.getSlideDeck().getSlides().size()); List attachments = Stream.of(mediaMessage.getSlideDeck().getSlides()) .filter(s -> s.hasImage() || s.hasVideo()) .map(Slide::asAttachment) .toList(); for (Attachment attachment : attachments) { Uri uri = attachment.getUri(); if (uri != null) { mediaList.add(new Media(uri, attachment.getContentType(), System.currentTimeMillis(), attachment.getWidth(), attachment.getHeight(), attachment.getSize(), 0, attachment.isBorderless(), Optional.absent(), Optional.fromNullable(attachment.getCaption()), Optional.absent())); } } if (!mediaList.isEmpty()) { composeIntent.putExtra(ConversationActivity.MEDIA_EXTRA, mediaList); } } else if (mediaMessage.containsMediaSlide()) { Slide slide = mediaMessage.getSlideDeck().getSlides().get(0); composeIntent.putExtra(Intent.EXTRA_STREAM, slide.getUri()); composeIntent.setType(slide.getContentType()); composeIntent.putExtra(ConversationActivity.BORDERLESS_EXTRA, slide.isBorderless()); if (slide.hasSticker()) { composeIntent.putExtra(ConversationActivity.STICKER_EXTRA, slide.asAttachment().getSticker()); composeIntent.setType(slide.asAttachment().getContentType()); } } if (mediaMessage.getSlideDeck().getTextSlide() != null && mediaMessage.getSlideDeck().getTextSlide().getUri() != null) { try (InputStream stream = PartAuthority.getAttachmentStream(requireContext(), mediaMessage.getSlideDeck().getTextSlide().getUri())) { String fullBody = Util.readFullyAsString(stream); composeIntent.putExtra(Intent.EXTRA_TEXT, fullBody); } catch (IOException e) { Log.w(TAG, "Failed to read long message text when forwarding."); } } } return composeIntent; }, this::startActivity); } private void handleResendMessage(final MessageRecord message) { final Context context = getActivity().getApplicationContext(); new AsyncTask() { @Override protected Void doInBackground(MessageRecord... messageRecords) { MessageSender.resend(context, messageRecords[0]); return null; } }.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, message); } private void handleReplyMessage(final ConversationMessage message) { if (getActivity() != null) { //noinspection ConstantConditions ((AppCompatActivity) getActivity()).getSupportActionBar().collapseActionView(); } listener.handleReplyMessage(message); } private void handleSaveAttachment(final MediaMmsMessageRecord message) { if (message.isViewOnce()) { throw new AssertionError("Cannot save a view-once message."); } SaveAttachmentTask.showWarningDialog(getActivity(), new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int which) { List attachments = Stream.of(message.getSlideDeck().getSlides()) .filter(s -> s.getUri() != null && (s.hasImage() || s.hasVideo() || s.hasAudio() || s.hasDocument())) .map(s -> new SaveAttachmentTask.Attachment(s.getUri(), s.getContentType(), message.getDateReceived(), s.getFileName().orNull())) .toList(); if (!Util.isEmpty(attachments)) { SaveAttachmentTask saveTask = new SaveAttachmentTask(getActivity()); saveTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, attachments.toArray(new SaveAttachmentTask.Attachment[0])); return; } Log.w(TAG, "No slide with attachable media found, failing nicely."); Toast.makeText(getActivity(), getResources().getQuantityString(R.plurals.ConversationFragment_error_while_saving_attachments_to_sd_card, 1), Toast.LENGTH_LONG).show(); } }); } private void clearHeaderIfNotTyping(ConversationAdapter adapter) { if (adapter.getHeaderView() != typingView) { adapter.setHeaderView(null); } } public long stageOutgoingMessage(OutgoingMediaMessage message) { MessageRecord messageRecord = MmsDatabase.readerFor(message, threadId).getCurrent(); if (getListAdapter() != null) { clearHeaderIfNotTyping(getListAdapter()); setLastSeen(0); getListAdapter().addFastRecord(ConversationMessageFactory.createWithResolvedData(messageRecord, messageRecord.getDisplayBody(requireContext()), message.getMentions())); list.post(() -> list.scrollToPosition(0)); } return messageRecord.getId(); } public long stageOutgoingMessage(OutgoingTextMessage message) { MessageRecord messageRecord = SmsDatabase.readerFor(message, threadId).getCurrent(); if (getListAdapter() != null) { clearHeaderIfNotTyping(getListAdapter()); setLastSeen(0); getListAdapter().addFastRecord(ConversationMessageFactory.createWithResolvedData(messageRecord)); list.post(() -> list.scrollToPosition(0)); } return messageRecord.getId(); } public void releaseOutgoingMessage(long id) { if (getListAdapter() != null) { getListAdapter().releaseFastRecord(id); } } private void presentConversationMetadata(@NonNull ConversationData conversation) { ConversationAdapter adapter = getListAdapter(); if (adapter == null) { return; } adapter.setFooterView(conversationBanner); Runnable afterScroll = () -> { if (!conversation.isMessageRequestAccepted()) { snapToTopDataObserver.requestScrollPosition(adapter.getItemCount() - 1); } setLastSeen(conversation.getLastSeen()); if (!conversation.hasPreMessageRequestMessages()) { clearHeaderIfNotTyping(adapter); } else { if (!conversation.hasSent() && !recipient.get().isSystemContact() && !recipient.get().isGroup() && recipient.get().getRegistered() == RecipientDatabase.RegisteredState.REGISTERED) { adapter.setHeaderView(unknownSenderView); } else { clearHeaderIfNotTyping(adapter); } } listener.onCursorChanged(); conversationScrollListener.onScrolled(list, 0, 0); }; int lastSeenPosition = adapter.getAdapterPositionForMessagePosition(conversation.getLastSeenPosition()); int lastScrolledPosition = adapter.getAdapterPositionForMessagePosition(conversation.getLastScrolledPosition()); if (conversation.getThreadSize() == 0) { afterScroll.run(); } else if (conversation.shouldJumpToMessage()) { snapToTopDataObserver.buildScrollPosition(conversation.getJumpToPosition()) .withOnScrollRequestComplete(() -> { afterScroll.run(); getListAdapter().pulseAtPosition(conversation.getJumpToPosition()); }) .submit(); } else if (conversation.isMessageRequestAccepted()) { snapToTopDataObserver.buildScrollPosition(conversation.shouldScrollToLastSeen() ? lastSeenPosition : lastScrolledPosition) .withOnPerformScroll((layoutManager, position) -> layoutManager.scrollToPositionWithOffset(position, list.getHeight())) .withOnScrollRequestComplete(afterScroll) .submit(); } else { snapToTopDataObserver.buildScrollPosition(adapter.getItemCount() - 1) .withOnScrollRequestComplete(afterScroll) .submit(); } } private boolean isAtBottom() { if (list.getChildCount() == 0) return true; int firstVisiblePosition = getListLayoutManager().findFirstVisibleItemPosition(); if (isTypingIndicatorShowing()) { RecyclerView.ViewHolder item1 = list.findViewHolderForAdapterPosition(1); return firstVisiblePosition <= 1 && item1 != null && item1.itemView.getBottom() <= list.getHeight(); } return firstVisiblePosition == 0 && list.getChildAt(0).getBottom() <= list.getHeight(); } private boolean isTypingIndicatorShowing() { return getListAdapter().getHeaderView() == typingView; } public void onSearchQueryUpdated(@Nullable String query) { if (getListAdapter() != null) { getListAdapter().onSearchQueryUpdated(query); } } @SuppressWarnings("CodeBlock2Expr") public void jumpToMessage(@NonNull RecipientId author, long timestamp, @Nullable Runnable onMessageNotFound) { SimpleTask.run(getLifecycle(), () -> { return DatabaseFactory.getMmsSmsDatabase(getContext()) .getMessagePositionInConversation(threadId, timestamp, author); }, p -> moveToPosition(p + (isTypingIndicatorShowing() ? 1 : 0), onMessageNotFound)); } private void moveToPosition(int position, @Nullable Runnable onMessageNotFound) { conversationViewModel.onConversationDataAvailable(threadId, position); snapToTopDataObserver.buildScrollPosition(position) .withOnPerformScroll(((layoutManager, p) -> list.post(() -> { if (Math.abs(layoutManager.findFirstVisibleItemPosition() - p) < SCROLL_ANIMATION_THRESHOLD) { View child = layoutManager.findViewByPosition(position); if (child != null && layoutManager.isViewPartiallyVisible(child, true, false)) { getListAdapter().pulseAtPosition(position); } else { pulsePosition = position; } list.smoothScrollToPosition(p); } else { layoutManager.scrollToPosition(p); getListAdapter().pulseAtPosition(position); } }) )) .withOnInvalidPosition(() -> { if (onMessageNotFound != null) { onMessageNotFound.run(); } Log.w(TAG, "[moveToMentionPosition] Tried to navigate to mention, but it wasn't found."); }) .submit(); } private void maybeShowSwipeToReplyTooltip() { if (!TextSecurePreferences.hasSeenSwipeToReplyTooltip(requireContext())) { int text = getResources().getConfiguration().getLayoutDirection() == View.LAYOUT_DIRECTION_LTR ? R.string.ConversationFragment_you_can_swipe_to_the_right_reply : R.string.ConversationFragment_you_can_swipe_to_the_left_reply; TooltipPopup.forTarget(requireActivity().findViewById(R.id.menu_context_reply)) .setText(text) .setTextColor(getResources().getColor(R.color.core_white)) .setBackgroundTint(getResources().getColor(R.color.core_ultramarine)) .show(TooltipPopup.POSITION_BELOW); TextSecurePreferences.setHasSeenSwipeToReplyTooltip(requireContext(), true); } } private void initializeScrollButtonAnimations() { scrollButtonInAnimation = AnimationUtils.loadAnimation(requireContext(), R.anim.fade_scale_in); scrollButtonOutAnimation = AnimationUtils.loadAnimation(requireContext(), R.anim.fade_scale_out); mentionButtonInAnimation = AnimationUtils.loadAnimation(requireContext(), R.anim.fade_scale_in); mentionButtonOutAnimation = AnimationUtils.loadAnimation(requireContext(), R.anim.fade_scale_out); scrollButtonInAnimation.setDuration(100); scrollButtonOutAnimation.setDuration(50); mentionButtonInAnimation.setDuration(100); mentionButtonOutAnimation.setDuration(50); } private void scrollToNextMention() { SimpleTask.run(getViewLifecycleOwner().getLifecycle(), () -> { MessageDatabase mmsDatabase = DatabaseFactory.getMmsDatabase(ApplicationDependencies.getApplication()); return mmsDatabase.getOldestUnreadMentionDetails(threadId); }, (pair) -> { if (pair != null) { jumpToMessage(pair.first(), pair.second(), () -> {}); } }); } private void postMarkAsReadRequest() { if (getListAdapter().hasNoConversationMessages()) { return; } int position = getListLayoutManager().findFirstVisibleItemPosition(); if (position == getListAdapter().getItemCount() - 1) { return; } if (position >= (isTypingIndicatorShowing() ? 1 : 0)) { ConversationMessage item = getListAdapter().getItem(position); if (item != null) { long timestamp = item.getMessageRecord() .getDateReceived(); markReadHelper.onViewsRevealed(timestamp); } } } public interface ConversationFragmentListener { void setThreadId(long threadId); void handleReplyMessage(ConversationMessage conversationMessage); void onMessageActionToolbarOpened(); void onForwardClicked(); void onMessageRequest(@NonNull MessageRequestViewModel viewModel); void handleReaction(@NonNull View maskTarget, @NonNull MessageRecord messageRecord, @NonNull Toolbar.OnMenuItemClickListener toolbarListener, @NonNull ConversationReactionOverlay.OnHideListener onHideListener); void onCursorChanged(); void onListVerticalTranslationChanged(float translationY); void onMessageWithErrorClicked(@NonNull MessageRecord messageRecord); void handleReactionDetails(@NonNull View maskTarget); } private class ConversationScrollListener extends OnScrollListener { private final ConversationDateHeader conversationDateHeader; private boolean wasAtBottom = true; private long lastPositionId = -1; ConversationScrollListener(@NonNull Context context) { this.conversationDateHeader = new ConversationDateHeader(context, scrollDateHeader); } @Override public void onScrolled(@NonNull final RecyclerView rv, final int dx, final int dy) { boolean currentlyAtBottom = !rv.canScrollVertically(1); boolean currentlyAtZoomScrollHeight = isAtZoomScrollHeight(); int positionId = getHeaderPositionId(); if (currentlyAtBottom && !wasAtBottom) { ViewUtil.fadeOut(composeDivider, 50, View.INVISIBLE); } else if (!currentlyAtBottom && wasAtBottom) { ViewUtil.fadeIn(composeDivider, 500); } if (currentlyAtBottom) { conversationViewModel.setShowScrollButtons(false); } else if (currentlyAtZoomScrollHeight) { conversationViewModel.setShowScrollButtons(true); } if (positionId != lastPositionId) { bindScrollHeader(conversationDateHeader, positionId); } wasAtBottom = currentlyAtBottom; lastPositionId = positionId; postMarkAsReadRequest(); } @Override public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) { if (newState == RecyclerView.SCROLL_STATE_DRAGGING) { conversationDateHeader.show(); } else if (newState == RecyclerView.SCROLL_STATE_IDLE) { conversationDateHeader.hide(); if (pulsePosition != -1) { getListAdapter().pulseAtPosition(pulsePosition); pulsePosition = -1; } } } private boolean isAtZoomScrollHeight() { return getListLayoutManager().findFirstCompletelyVisibleItemPosition() > 4; } private int getHeaderPositionId() { return getListLayoutManager().findLastVisibleItemPosition(); } private void bindScrollHeader(StickyHeaderViewHolder headerViewHolder, int positionId) { if (((ConversationAdapter)list.getAdapter()).getHeaderId(positionId) != -1) { ((ConversationAdapter) list.getAdapter()).onBindHeaderViewHolder(headerViewHolder, positionId); } } } private class ConversationFragmentItemClickListener implements ItemClickListener { @Override public void onItemClick(ConversationMessage conversationMessage) { if (actionMode != null) { ((ConversationAdapter) list.getAdapter()).toggleSelection(conversationMessage); list.getAdapter().notifyDataSetChanged(); if (getListAdapter().getSelectedItems().size() == 0) { actionMode.finish(); } else { setCorrectMenuVisibility(actionMode.getMenu()); actionMode.setTitle(String.valueOf(getListAdapter().getSelectedItems().size())); } } } @Override public void onItemLongClick(View maskTarget, ConversationMessage conversationMessage) { if (actionMode != null) return; MessageRecord messageRecord = conversationMessage.getMessageRecord(); if (messageRecord.isSecure() && !messageRecord.isRemoteDelete() && !messageRecord.isUpdate() && !recipient.get().isBlocked() && !messageRequestViewModel.shouldShowMessageRequest() && (!recipient.get().isGroup() || recipient.get().isActiveGroup()) && ((ConversationAdapter) list.getAdapter()).getSelectedItems().isEmpty()) { isReacting = true; list.setLayoutFrozen(true); listener.handleReaction(maskTarget, messageRecord, new ReactionsToolbarListener(conversationMessage), () -> { isReacting = false; list.setLayoutFrozen(false); }); } else { ((ConversationAdapter) list.getAdapter()).toggleSelection(conversationMessage); list.getAdapter().notifyDataSetChanged(); actionMode = ((AppCompatActivity)getActivity()).startSupportActionMode(actionModeCallback); } } @Override public void onQuoteClicked(MmsMessageRecord messageRecord) { if (messageRecord.getQuote() == null) { Log.w(TAG, "Received a 'quote clicked' event, but there's no quote..."); return; } if (messageRecord.getQuote().isOriginalMissing()) { Log.i(TAG, "Clicked on a quote whose original message we never had."); Toast.makeText(getContext(), R.string.ConversationFragment_quoted_message_not_found, Toast.LENGTH_SHORT).show(); return; } SimpleTask.run(getLifecycle(), () -> { return DatabaseFactory.getMmsSmsDatabase(getContext()) .getQuotedMessagePosition(threadId, messageRecord.getQuote().getId(), messageRecord.getQuote().getAuthor()); }, p -> moveToPosition(p + (isTypingIndicatorShowing() ? 1 : 0), () -> { Toast.makeText(getContext(), R.string.ConversationFragment_quoted_message_no_longer_available, Toast.LENGTH_SHORT).show(); })); } @Override public void onLinkPreviewClicked(@NonNull LinkPreview linkPreview) { if (getContext() != null && getActivity() != null) { CommunicationActions.openBrowserLink(getActivity(), linkPreview.getUrl()); } } @Override public void onMoreTextClicked(@NonNull RecipientId conversationRecipientId, long messageId, boolean isMms) { if (getContext() != null && getActivity() != null) { startActivity(LongMessageActivity.getIntent(getContext(), conversationRecipientId, messageId, isMms)); } } @Override public void onStickerClicked(@NonNull StickerLocator sticker) { if (getContext() != null && getActivity() != null) { startActivity(StickerPackPreviewActivity.getIntent(sticker.getPackId(), sticker.getPackKey())); } } @Override public void onViewOnceMessageClicked(@NonNull MmsMessageRecord messageRecord) { if (!messageRecord.isViewOnce()) { throw new AssertionError("Non-revealable message clicked."); } if (!ViewOnceUtil.isViewable(messageRecord)) { int stringRes = messageRecord.isOutgoing() ? R.string.ConversationFragment_outgoing_view_once_media_files_are_automatically_removed : R.string.ConversationFragment_you_already_viewed_this_message; Toast.makeText(requireContext(), stringRes, Toast.LENGTH_SHORT).show(); return; } SimpleTask.run(getLifecycle(), () -> { Log.i(TAG, "Copying the view-once photo to temp storage and deleting underlying media."); try { Slide thumbnailSlide = messageRecord.getSlideDeck().getThumbnailSlide(); InputStream inputStream = PartAuthority.getAttachmentStream(requireContext(), thumbnailSlide.getUri()); Uri tempUri = BlobProvider.getInstance().forData(inputStream, thumbnailSlide.getFileSize()) .withMimeType(thumbnailSlide.getContentType()) .createForSingleSessionOnDisk(requireContext()); DatabaseFactory.getAttachmentDatabase(requireContext()).deleteAttachmentFilesForViewOnceMessage(messageRecord.getId()); ApplicationContext.getInstance(requireContext()) .getViewOnceMessageManager() .scheduleIfNecessary(); ApplicationDependencies.getJobManager().add(new MultiDeviceViewOnceOpenJob(new MessageDatabase.SyncMessageId(messageRecord.getIndividualRecipient().getId(), messageRecord.getDateSent()))); return tempUri; } catch (IOException e) { return null; } }, (uri) -> { if (uri != null) { startActivity(ViewOnceMessageActivity.getIntent(requireContext(), messageRecord.getId(), uri)); } else { Log.w(TAG, "Failed to open view-once photo. Showing a toast and deleting the attachments for the message just in case."); Toast.makeText(requireContext(), R.string.ConversationFragment_failed_to_open_message, Toast.LENGTH_SHORT).show(); SignalExecutors.BOUNDED.execute(() -> DatabaseFactory.getAttachmentDatabase(requireContext()).deleteAttachmentFilesForViewOnceMessage(messageRecord.getId())); } }); } @Override public void onSharedContactDetailsClicked(@NonNull Contact contact, @NonNull View avatarTransitionView) { if (getContext() != null && getActivity() != null) { Bundle bundle = ActivityOptionsCompat.makeSceneTransitionAnimation(getActivity(), avatarTransitionView, "avatar").toBundle(); ActivityCompat.startActivity(getActivity(), SharedContactDetailsActivity.getIntent(getContext(), contact), bundle); } } @Override public void onAddToContactsClicked(@NonNull Contact contactWithAvatar) { if (getContext() != null) { new AsyncTask() { @Override protected Intent doInBackground(Void... voids) { return ContactUtil.buildAddToContactsIntent(getContext(), contactWithAvatar); } @Override protected void onPostExecute(Intent intent) { startActivityForResult(intent, CODE_ADD_EDIT_CONTACT); } }.execute(); } } @Override public void onMessageSharedContactClicked(@NonNull List choices) { if (getContext() == null) return; ContactUtil.selectRecipientThroughDialog(getContext(), choices, locale, recipient -> { CommunicationActions.startConversation(getContext(), recipient, null); }); } @Override public void onInviteSharedContactClicked(@NonNull List choices) { if (getContext() == null) return; ContactUtil.selectRecipientThroughDialog(getContext(), choices, locale, recipient -> { CommunicationActions.composeSmsThroughDefaultApp(getContext(), recipient, getString(R.string.InviteActivity_lets_switch_to_signal, getString(R.string.install_url))); }); } @Override public void onReactionClicked(@NonNull View reactionTarget, long messageId, boolean isMms) { if (getContext() == null) return; listener.handleReactionDetails(reactionTarget); ReactionsBottomSheetDialogFragment.create(messageId, isMms).show(requireFragmentManager(), null); } @Override public void onGroupMemberClicked(@NonNull RecipientId recipientId, @NonNull GroupId groupId) { if (getContext() == null) return; RecipientBottomSheetDialogFragment.create(recipientId, groupId).show(requireFragmentManager(), "BOTTOM"); } @Override public void onMessageWithErrorClicked(@NonNull MessageRecord messageRecord) { listener.onMessageWithErrorClicked(messageRecord); } @Override public boolean onUrlClicked(@NonNull String url) { return CommunicationActions.handlePotentialGroupLinkUrl(requireActivity(), url); } } @Override public void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == CODE_ADD_EDIT_CONTACT && getContext() != null) { ApplicationDependencies.getJobManager().add(new DirectoryRefreshJob(false)); } } private void handleEnterMultiSelect(@NonNull ConversationMessage conversationMessage) { ((ConversationAdapter) list.getAdapter()).toggleSelection(conversationMessage); list.getAdapter().notifyDataSetChanged(); actionMode = ((AppCompatActivity)getActivity()).startSupportActionMode(actionModeCallback); } private final class ConversationSnapToTopDataObserver extends SnapToTopDataObserver { public ConversationSnapToTopDataObserver(@NonNull RecyclerView recyclerView, @Nullable ScrollRequestValidator scrollRequestValidator) { super(recyclerView, scrollRequestValidator, () -> { list.scrollToPosition(0); list.post(ConversationFragment.this::postMarkAsReadRequest); }); } @Override public void onItemRangeMoved(int fromPosition, int toPosition, int itemCount) { // Do nothing. } @Override public void onItemRangeInserted(int positionStart, int itemCount) { if (positionStart == 0 && itemCount == 1 && isTypingIndicatorShowing()) { return; } super.onItemRangeInserted(positionStart, itemCount); } } private final class ConversationScrollRequestValidator implements SnapToTopDataObserver.ScrollRequestValidator { @Override public boolean isPositionStillValid(int position) { if (getListAdapter() == null) { return position >= 0; } else { return position >= 0 && position < getListAdapter().getItemCount(); } } @Override public boolean isItemAtPositionLoaded(int position) { if (getListAdapter() == null) { return false; } else if (getListAdapter().hasFooter() && position == getListAdapter().getItemCount() - 1) { return true; } else { return getListAdapter().getItem(position) != null; } } } private class ReactionsToolbarListener implements Toolbar.OnMenuItemClickListener { private final ConversationMessage conversationMessage; private ReactionsToolbarListener(@NonNull ConversationMessage conversationMessage) { this.conversationMessage = conversationMessage; } @Override public boolean onMenuItemClick(MenuItem item) { switch (item.getItemId()) { case R.id.action_info: handleDisplayDetails(conversationMessage); return true; case R.id.action_delete: handleDeleteMessages(Sets.newHashSet(conversationMessage)); return true; case R.id.action_copy: handleCopyMessage(Sets.newHashSet(conversationMessage)); return true; case R.id.action_reply: handleReplyMessage(conversationMessage); return true; case R.id.action_multiselect: handleEnterMultiSelect(conversationMessage); return true; case R.id.action_forward: handleForwardMessage(conversationMessage); return true; case R.id.action_download: handleSaveAttachment((MediaMmsMessageRecord) conversationMessage.getMessageRecord()); return true; default: return false; } } } private class ActionModeCallback implements ActionMode.Callback { private int statusBarColor; @Override public boolean onCreateActionMode(ActionMode mode, Menu menu) { MenuInflater inflater = mode.getMenuInflater(); inflater.inflate(R.menu.conversation_context, menu); mode.setTitle("1"); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { Window window = getActivity().getWindow(); statusBarColor = window.getStatusBarColor(); window.setStatusBarColor(getResources().getColor(R.color.action_mode_status_bar)); } setCorrectMenuVisibility(menu); AdaptiveActionsToolbar.adjustMenuActions(menu, 10, requireActivity().getWindow().getDecorView().getMeasuredWidth()); listener.onMessageActionToolbarOpened(); return true; } @Override public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) { return false; } @Override public void onDestroyActionMode(ActionMode mode) { ((ConversationAdapter)list.getAdapter()).clearSelection(); list.getAdapter().notifyDataSetChanged(); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { getActivity().getWindow().setStatusBarColor(statusBarColor); } actionMode = null; } @Override public boolean onActionItemClicked(ActionMode mode, MenuItem item) { if (actionMode == null) return false; switch(item.getItemId()) { case R.id.menu_context_copy: handleCopyMessage(getListAdapter().getSelectedItems()); actionMode.finish(); return true; case R.id.menu_context_delete_message: handleDeleteMessages(getListAdapter().getSelectedItems()); actionMode.finish(); return true; case R.id.menu_context_details: handleDisplayDetails(getSelectedConversationMessage()); actionMode.finish(); return true; case R.id.menu_context_forward: handleForwardMessage(getSelectedConversationMessage()); actionMode.finish(); return true; case R.id.menu_context_resend: handleResendMessage(getSelectedConversationMessage().getMessageRecord()); actionMode.finish(); return true; case R.id.menu_context_save_attachment: handleSaveAttachment((MediaMmsMessageRecord) getSelectedConversationMessage().getMessageRecord()); actionMode.finish(); return true; case R.id.menu_context_reply: maybeShowSwipeToReplyTooltip(); handleReplyMessage(getSelectedConversationMessage()); actionMode.finish(); return true; } return false; } } private static class ConversationDateHeader extends StickyHeaderViewHolder { private final Animation animateIn; private final Animation animateOut; private boolean pendingHide = false; private ConversationDateHeader(Context context, TextView textView) { super(textView); this.animateIn = AnimationUtils.loadAnimation(context, R.anim.slide_from_top); this.animateOut = AnimationUtils.loadAnimation(context, R.anim.slide_to_top); this.animateIn.setDuration(100); this.animateOut.setDuration(100); } public void show() { if (textView.getText() == null || textView.getText().length() == 0) { return; } if (pendingHide) { pendingHide = false; } else { ViewUtil.animateIn(textView, animateIn); } } public void hide() { pendingHide = true; textView.postDelayed(new Runnable() { @Override public void run() { if (pendingHide) { pendingHide = false; ViewUtil.animateOut(textView, animateOut, View.GONE); } } }, 400); } } }