/* * 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; import android.Manifest; import android.annotation.SuppressLint; import android.content.Context; import android.database.Cursor; import android.os.AsyncTask; import android.os.Bundle; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.view.WindowManager; import android.widget.Button; import android.widget.TextView; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.appcompat.app.AlertDialog; import androidx.fragment.app.Fragment; import androidx.loader.app.LoaderManager; import androidx.loader.content.Loader; import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; import androidx.swiperefreshlayout.widget.SwipeRefreshLayout; import com.pnikosis.materialishprogress.ProgressWheel; import org.thoughtcrime.securesms.components.RecyclerViewFastScroller; import org.thoughtcrime.securesms.contacts.ContactSelectionListAdapter; import org.thoughtcrime.securesms.contacts.ContactSelectionListItem; import org.thoughtcrime.securesms.contacts.ContactsCursorLoader; import org.thoughtcrime.securesms.contacts.ContactsCursorLoader.DisplayMode; import org.thoughtcrime.securesms.contacts.SelectedContact; import org.thoughtcrime.securesms.logging.Log; import org.thoughtcrime.securesms.mms.GlideApp; import org.thoughtcrime.securesms.permissions.Permissions; import org.thoughtcrime.securesms.contacts.sync.DirectoryHelper; import org.thoughtcrime.securesms.recipients.Recipient; import org.thoughtcrime.securesms.recipients.RecipientId; import org.thoughtcrime.securesms.util.StickyHeaderDecoration; import org.thoughtcrime.securesms.util.TextSecurePreferences; import org.thoughtcrime.securesms.util.UsernameUtil; import org.thoughtcrime.securesms.util.ViewUtil; import org.thoughtcrime.securesms.util.adapter.FixedViewsAdapter; import org.thoughtcrime.securesms.util.adapter.RecyclerViewConcatenateAdapterStickyHeader; import org.thoughtcrime.securesms.util.concurrent.SimpleTask; import org.thoughtcrime.securesms.util.views.SimpleProgressDialog; import org.whispersystems.libsignal.util.guava.Optional; import java.io.IOException; import java.util.LinkedList; import java.util.List; import java.util.Set; /** * Fragment for selecting a one or more contacts from a list. * * @author Moxie Marlinspike * */ public final class ContactSelectionListFragment extends Fragment implements LoaderManager.LoaderCallbacks { @SuppressWarnings("unused") private static final String TAG = Log.tag(ContactSelectionListFragment.class); public static final String DISPLAY_MODE = "display_mode"; public static final String MULTI_SELECT = "multi_select"; public static final String REFRESHABLE = "refreshable"; public static final String RECENTS = "recents"; private TextView emptyText; private Set selectedContacts; private OnContactSelectedListener onContactSelectedListener; private SwipeRefreshLayout swipeRefresh; private View showContactsLayout; private Button showContactsButton; private TextView showContactsDescription; private ProgressWheel showContactsProgress; private String cursorFilter; private RecyclerView recyclerView; private RecyclerViewFastScroller fastScroller; private ContactSelectionListAdapter cursorRecyclerViewAdapter; @Nullable private FixedViewsAdapter footerAdapter; @Nullable private InviteCallback inviteCallback; @Override public void onAttach(@NonNull Context context) { super.onAttach(context); if (context instanceof InviteCallback) { inviteCallback = (InviteCallback) context; } } @Override public void onActivityCreated(Bundle icicle) { super.onActivityCreated(icicle); initializeCursor(); } @Override public void onStart() { super.onStart(); Permissions.with(this) .request(Manifest.permission.WRITE_CONTACTS, Manifest.permission.READ_CONTACTS) .ifNecessary() .onAllGranted(() -> { if (!TextSecurePreferences.hasSuccessfullyRetrievedDirectory(getActivity())) { handleContactPermissionGranted(); } else { this.getLoaderManager().initLoader(0, null, this); } }) .onAnyDenied(() -> { getActivity().getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN); if (getActivity().getIntent().getBooleanExtra(RECENTS, false)) { getLoaderManager().initLoader(0, null, ContactSelectionListFragment.this); } else { initializeNoContactsPermission(); } }) .execute(); } @Override public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = inflater.inflate(R.layout.contact_selection_list_fragment, container, false); emptyText = ViewUtil.findById(view, android.R.id.empty); recyclerView = ViewUtil.findById(view, R.id.recycler_view); swipeRefresh = ViewUtil.findById(view, R.id.swipe_refresh); fastScroller = ViewUtil.findById(view, R.id.fast_scroller); showContactsLayout = view.findViewById(R.id.show_contacts_container); showContactsButton = view.findViewById(R.id.show_contacts_button); showContactsDescription = view.findViewById(R.id.show_contacts_description); showContactsProgress = view.findViewById(R.id.progress); recyclerView.setLayoutManager(new LinearLayoutManager(getActivity())); swipeRefresh.setEnabled(getActivity().getIntent().getBooleanExtra(REFRESHABLE, true)); return view; } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { Permissions.onRequestPermissionsResult(this, requestCode, permissions, grantResults); } public @NonNull List getSelectedContacts() { List selected = new LinkedList<>(); if (selectedContacts != null) { selected.addAll(selectedContacts); } return selected; } private boolean isMulti() { return getActivity().getIntent().getBooleanExtra(MULTI_SELECT, false); } private void initializeCursor() { cursorRecyclerViewAdapter = new ContactSelectionListAdapter(requireContext(), GlideApp.with(this), null, new ListClickListener(), isMulti()); selectedContacts = cursorRecyclerViewAdapter.getSelectedContacts(); RecyclerViewConcatenateAdapterStickyHeader concatenateAdapter = new RecyclerViewConcatenateAdapterStickyHeader(); concatenateAdapter.addAdapter(cursorRecyclerViewAdapter); if (inviteCallback != null) { footerAdapter = new FixedViewsAdapter(createInviteActionView(inviteCallback)); footerAdapter.hide(); concatenateAdapter.addAdapter(footerAdapter); } recyclerView.setAdapter(concatenateAdapter); recyclerView.addItemDecoration(new StickyHeaderDecoration(concatenateAdapter, true, true)); } private View createInviteActionView(@NonNull InviteCallback inviteCallback) { View view = LayoutInflater.from(requireContext()) .inflate(R.layout.contact_selection_invite_action_item, (ViewGroup) requireView(), false); view.setOnClickListener(v -> inviteCallback.onInvite()); return view; } private void initializeNoContactsPermission() { swipeRefresh.setVisibility(View.GONE); showContactsLayout.setVisibility(View.VISIBLE); showContactsProgress.setVisibility(View.INVISIBLE); showContactsDescription.setText(R.string.contact_selection_list_fragment__signal_needs_access_to_your_contacts_in_order_to_display_them); showContactsButton.setVisibility(View.VISIBLE); showContactsButton.setOnClickListener(v -> { Permissions.with(this) .request(Manifest.permission.WRITE_CONTACTS, Manifest.permission.READ_CONTACTS) .ifNecessary() .withPermanentDenialDialog(getString(R.string.ContactSelectionListFragment_signal_requires_the_contacts_permission_in_order_to_display_your_contacts)) .onSomeGranted(permissions -> { if (permissions.contains(Manifest.permission.WRITE_CONTACTS)) { handleContactPermissionGranted(); } }) .execute(); }); } public void setQueryFilter(String filter) { this.cursorFilter = filter; LoaderManager.getInstance(this).restartLoader(0, null, this); } public void resetQueryFilter() { setQueryFilter(null); swipeRefresh.setRefreshing(false); } public void setRefreshing(boolean refreshing) { swipeRefresh.setRefreshing(refreshing); } public void reset() { selectedContacts.clear(); if (!isDetached() && !isRemoving() && getActivity() != null && !getActivity().isFinishing()) { getLoaderManager().restartLoader(0, null, this); } } @Override public @NonNull Loader onCreateLoader(int id, Bundle args) { return new ContactsCursorLoader(getActivity(), getActivity().getIntent().getIntExtra(DISPLAY_MODE, DisplayMode.FLAG_ALL), cursorFilter, getActivity().getIntent().getBooleanExtra(RECENTS, false)); } @Override public void onLoadFinished(@NonNull Loader loader, @Nullable Cursor data) { swipeRefresh.setVisibility(View.VISIBLE); showContactsLayout.setVisibility(View.GONE); cursorRecyclerViewAdapter.changeCursor(data); if (footerAdapter != null) { footerAdapter.show(); } emptyText.setText(R.string.contact_selection_group_activity__no_contacts); boolean useFastScroller = data != null && data.getCount() > 20; recyclerView.setVerticalScrollBarEnabled(!useFastScroller); if (useFastScroller) { fastScroller.setVisibility(View.VISIBLE); fastScroller.setRecyclerView(recyclerView); } else { fastScroller.setRecyclerView(null); fastScroller.setVisibility(View.GONE); } } @Override public void onLoaderReset(@NonNull Loader loader) { cursorRecyclerViewAdapter.changeCursor(null); fastScroller.setVisibility(View.GONE); } @SuppressLint("StaticFieldLeak") private void handleContactPermissionGranted() { final Context context = requireContext(); new AsyncTask() { @Override protected void onPreExecute() { swipeRefresh.setVisibility(View.GONE); showContactsLayout.setVisibility(View.VISIBLE); showContactsButton.setVisibility(View.INVISIBLE); showContactsDescription.setText(R.string.ConversationListFragment_loading); showContactsProgress.setVisibility(View.VISIBLE); showContactsProgress.spin(); } @Override protected Boolean doInBackground(Void... voids) { try { DirectoryHelper.refreshDirectory(context, false); return true; } catch (IOException e) { Log.w(TAG, e); } return false; } @Override protected void onPostExecute(Boolean result) { if (result) { showContactsLayout.setVisibility(View.GONE); swipeRefresh.setVisibility(View.VISIBLE); reset(); } else { Toast.makeText(getContext(), R.string.ContactSelectionListFragment_error_retrieving_contacts_check_your_network_connection, Toast.LENGTH_LONG).show(); initializeNoContactsPermission(); } } }.execute(); } private class ListClickListener implements ContactSelectionListAdapter.ItemClickListener { @Override public void onItemClick(ContactSelectionListItem contact) { SelectedContact selectedContact = contact.isUsernameType() ? SelectedContact.forUsername(contact.getRecipientId().orNull(), contact.getNumber()) : SelectedContact.forPhone(contact.getRecipientId().orNull(), contact.getNumber()); if (!isMulti() || !selectedContacts.contains(selectedContact)) { if (contact.isUsernameType()) { AlertDialog loadingDialog = SimpleProgressDialog.show(requireContext()); SimpleTask.run(getViewLifecycleOwner().getLifecycle(), () -> { return UsernameUtil.fetchUuidForUsername(requireContext(), contact.getNumber()); }, uuid -> { loadingDialog.dismiss(); if (uuid.isPresent()) { Recipient recipient = Recipient.externalUsername(requireContext(), uuid.get(), contact.getNumber()); selectedContacts.add(SelectedContact.forUsername(recipient.getId(), contact.getNumber())); contact.setChecked(true); if (onContactSelectedListener != null) { onContactSelectedListener.onContactSelected(Optional.of(recipient.getId()), null); } } else { new AlertDialog.Builder(requireContext()) .setTitle(R.string.ContactSelectionListFragment_username_not_found) .setMessage(getString(R.string.ContactSelectionListFragment_s_is_not_a_signal_user, contact.getNumber())) .setPositiveButton(R.string.ContactSelectionListFragment_okay, (dialog, which) -> dialog.dismiss()) .show(); } }); } else { selectedContacts.add(selectedContact); contact.setChecked(true); if (onContactSelectedListener != null) { onContactSelectedListener.onContactSelected(contact.getRecipientId(), contact.getNumber()); } } } else { selectedContacts.remove(selectedContact); contact.setChecked(false); if (onContactSelectedListener != null) { onContactSelectedListener.onContactDeselected(contact.getRecipientId(), contact.getNumber()); } } } } public void setOnContactSelectedListener(OnContactSelectedListener onContactSelectedListener) { this.onContactSelectedListener = onContactSelectedListener; } public void setOnRefreshListener(SwipeRefreshLayout.OnRefreshListener onRefreshListener) { this.swipeRefresh.setOnRefreshListener(onRefreshListener); } public interface OnContactSelectedListener { void onContactSelected(Optional recipientId, String number); void onContactDeselected(Optional recipientId, String number); } public interface InviteCallback { void onInvite(); } }