/* * Copyright (c) 2018, Daniel Gultsch All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation and/or * other materials provided with the distribution. * * 3. Neither the name of the copyright holder nor the names of its contributors * may be used to endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package de.pixart.messenger.ui; import android.os.Bundle; import android.text.Editable; import android.text.InputType; import android.text.TextWatcher; import android.util.Log; import android.view.ContextMenu; import android.view.Menu; import android.view.MenuItem; import android.view.View; import android.widget.AdapterView; import android.widget.EditText; import androidx.appcompat.widget.Toolbar; import androidx.databinding.DataBindingUtil; import java.lang.ref.WeakReference; import java.util.ArrayList; import java.util.List; import de.pixart.messenger.Config; import de.pixart.messenger.R; import de.pixart.messenger.databinding.ActivitySearchBinding; import de.pixart.messenger.entities.Contact; import de.pixart.messenger.entities.Conversation; import de.pixart.messenger.entities.Conversational; import de.pixart.messenger.entities.Message; import de.pixart.messenger.services.MessageSearchTask; import de.pixart.messenger.ui.adapter.MessageAdapter; import de.pixart.messenger.ui.interfaces.OnSearchResultsAvailable; import de.pixart.messenger.ui.util.ChangeWatcher; import de.pixart.messenger.ui.util.DateSeparator; import de.pixart.messenger.ui.util.ListViewUtils; import de.pixart.messenger.ui.util.PendingItem; import de.pixart.messenger.ui.util.ShareUtil; import de.pixart.messenger.ui.util.StyledAttributes; import de.pixart.messenger.utils.FtsUtils; import de.pixart.messenger.utils.MessageUtils; import de.pixart.messenger.utils.UIHelper; import static de.pixart.messenger.ui.util.SoftKeyboardUtils.hideSoftKeyboard; import static de.pixart.messenger.ui.util.SoftKeyboardUtils.showKeyboard; public class SearchActivity extends XmppActivity implements TextWatcher, OnSearchResultsAvailable, MessageAdapter.OnContactPictureClicked { private static final String EXTRA_SEARCH_TERM = "search-term"; private ActivitySearchBinding binding; private MessageAdapter messageListAdapter; private final List messages = new ArrayList<>(); private WeakReference selectedMessageReference = new WeakReference<>(null); private final ChangeWatcher> currentSearch = new ChangeWatcher<>(); private final PendingItem pendingSearchTerm = new PendingItem<>(); private final PendingItem> pendingSearch = new PendingItem<>(); @Override public void onCreate(final Bundle bundle) { final String searchTerm = bundle == null ? null : bundle.getString(EXTRA_SEARCH_TERM); if (searchTerm != null) { pendingSearchTerm.push(searchTerm); } super.onCreate(bundle); this.binding = DataBindingUtil.setContentView(this, R.layout.activity_search); setSupportActionBar((Toolbar) this.binding.toolbar); configureActionBar(getSupportActionBar()); this.messageListAdapter = new MessageAdapter(this, this.messages); this.messageListAdapter.setOnContactPictureClicked(this); this.binding.searchResults.setAdapter(messageListAdapter); registerForContextMenu(this.binding.searchResults); } @Override public boolean onCreateOptionsMenu(final Menu menu) { getMenuInflater().inflate(R.menu.activity_search, menu); final MenuItem searchActionMenuItem = menu.findItem(R.id.action_search); final EditText searchField = searchActionMenuItem.getActionView().findViewById(R.id.search_field); final String term = pendingSearchTerm.pop(); if (term != null) { searchField.append(term); List searchTerm = FtsUtils.parse(term); if (xmppConnectionService != null) { if (currentSearch.watch(searchTerm)) { xmppConnectionService.search(searchTerm, this); } } else { pendingSearch.push(searchTerm); } } searchField.addTextChangedListener(this); searchField.setHint(R.string.search_messages); searchField.setInputType(InputType.TYPE_CLASS_TEXT | InputType.TYPE_TEXT_FLAG_AUTO_COMPLETE); showKeyboard(searchField); return super.onCreateOptionsMenu(menu); } @Override public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) { AdapterView.AdapterContextMenuInfo acmi = (AdapterView.AdapterContextMenuInfo) menuInfo; final Message message = this.messages.get(acmi.position); this.selectedMessageReference = new WeakReference<>(message); getMenuInflater().inflate(R.menu.search_result_context, menu); MenuItem copy = menu.findItem(R.id.copy_message); MenuItem quote = menu.findItem(R.id.quote_message); MenuItem copyUrl = menu.findItem(R.id.copy_url); if (message.isGeoUri()) { copy.setVisible(false); quote.setVisible(false); } else { copyUrl.setVisible(false); } super.onCreateContextMenu(menu, v, menuInfo); } @Override public boolean onOptionsItemSelected(MenuItem item) { if (item.getItemId() == android.R.id.home) { hideSoftKeyboard(this); } return super.onOptionsItemSelected(item); } @Override public boolean onContextItemSelected(MenuItem item) { final Message message = selectedMessageReference.get(); final boolean multi = message.getConversation().getMode() == Conversational.MODE_MULTI; final String user = multi ? UIHelper.getDisplayedMucCounterpart(message.getCounterpart()) : null; if (message != null) { switch (item.getItemId()) { case R.id.open_conversation: switchToConversation(wrap(message.getConversation())); break; case R.id.share_with: ShareUtil.share(this, message, user); break; case R.id.copy_message: ShareUtil.copyToClipboard(this, message); break; case R.id.copy_url: ShareUtil.copyUrlToClipboard(this, message); break; case R.id.quote_message: quote(message, user); break; } } return super.onContextItemSelected(item); } @Override public void onSaveInstanceState(Bundle bundle) { List term = currentSearch.get(); if (term != null && term.size() > 0) { bundle.putString(EXTRA_SEARCH_TERM, FtsUtils.toUserEnteredString(term)); } super.onSaveInstanceState(bundle); } private void quote(Message message, String user) { Log.d(Config.LOGTAG, "Quote User: " + user); switchToConversationAndQuote(wrap(message.getConversation()), MessageUtils.prepareQuote(message), user); } private Conversation wrap(Conversational conversational) { if (conversational instanceof Conversation) { return (Conversation) conversational; } else { return xmppConnectionService.findOrCreateConversation(conversational.getAccount(), conversational.getJid(), conversational.getMode() == Conversational.MODE_MULTI, true, true); } } @Override protected void refreshUiReal() { } @Override void onBackendConnected() { final List searchTerm = pendingSearch.pop(); if (searchTerm != null && currentSearch.watch(searchTerm)) { xmppConnectionService.search(searchTerm, this); } } private void changeBackground(boolean hasSearch, boolean hasResults) { if (hasSearch) { if (hasResults) { binding.searchResults.setBackgroundColor(StyledAttributes.getColor(this, R.attr.color_background_tertiary)); } else { binding.searchResults.setBackground(StyledAttributes.getDrawable(this, R.attr.activity_background_no_results)); } } else { binding.searchResults.setBackground(StyledAttributes.getDrawable(this, R.attr.activity_background_search)); } } @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { } @Override public void afterTextChanged(Editable s) { final List term = FtsUtils.parse(s.toString().trim()); if (!currentSearch.watch(term)) { return; } if (term.size() > 0) { xmppConnectionService.search(term, this); } else { MessageSearchTask.cancelRunningTasks(); this.messages.clear(); messageListAdapter.setHighlightedTerm(null); messageListAdapter.notifyDataSetChanged(); changeBackground(false, false); } } @Override public void onSearchResultsAvailable(List term, List messages) { runOnUiThread(() -> { this.messages.clear(); messageListAdapter.setHighlightedTerm(term); DateSeparator.addAll(messages); this.messages.addAll(messages); messageListAdapter.notifyDataSetChanged(); changeBackground(true, messages.size() > 0); ListViewUtils.scrollToBottom(this.binding.searchResults); }); } @Override public void onContactPictureClicked(Message message) { String fingerprint; if (message.getEncryption() == Message.ENCRYPTION_PGP || message.getEncryption() == Message.ENCRYPTION_DECRYPTED) { fingerprint = "pgp"; } else { fingerprint = message.getFingerprint(); } if (message.getStatus() == Message.STATUS_RECEIVED) { final Contact contact = message.getContact(); if (contact != null) { if (contact.isSelf()) { switchToAccount(message.getConversation().getAccount(), fingerprint); } else { switchToContactDetails(contact, fingerprint); } } } else { switchToAccount(message.getConversation().getAccount(), fingerprint); } } }