package eu.faircode.email; /* This file is part of FairEmail. FairEmail 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. FairEmail 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 FairEmail. If not, see . Copyright 2018-2024 by Marcel Bokhorst (M66B) */ import android.Manifest; import android.content.ClipData; import android.content.ClipboardManager; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.res.ColorStateList; import android.graphics.BitmapFactory; import android.graphics.Rect; import android.graphics.Typeface; import android.net.Uri; import android.os.Build; import android.os.Bundle; import android.provider.ContactsContract; import android.text.SpannableString; import android.text.TextUtils; import android.text.style.RelativeSizeSpan; import android.text.style.StyleSpan; import android.view.LayoutInflater; import android.view.Menu; import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; import android.widget.ImageView; import android.widget.TextView; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.appcompat.widget.PopupMenu; import androidx.core.content.pm.ShortcutInfoCompat; import androidx.fragment.app.Fragment; import androidx.lifecycle.Lifecycle; import androidx.lifecycle.LifecycleObserver; import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.OnLifecycleEvent; import androidx.recyclerview.widget.DiffUtil; import androidx.recyclerview.widget.ListUpdateCallback; import androidx.recyclerview.widget.RecyclerView; import java.io.InputStream; import java.io.UnsupportedEncodingException; import java.nio.charset.StandardCharsets; import java.text.NumberFormat; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import javax.mail.Address; import javax.mail.internet.InternetAddress; public class AdapterContact extends RecyclerView.Adapter { private Fragment parentFragment; private Context context; private LifecycleOwner owner; private LayoutInflater inflater; private boolean contacts; private int colorAccent; private int textColorSecondary; private String search = null; private Long account; private List types = new ArrayList<>(); private List all = new ArrayList<>(); private List selected = new ArrayList<>(); private NumberFormat NF = NumberFormat.getNumberInstance(); public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener, View.OnLongClickListener { private View view; private ImageView ivType; private ImageView ivAvatar; private TextView tvName; private TextView tvEmail; private TextView tvTimes; private TextView tvLast; private ImageView ivFavorite; private TwoStateOwner powner = new TwoStateOwner(owner, "ContactPopup"); ViewHolder(View itemView) { super(itemView); view = itemView.findViewById(R.id.clItem); ivType = itemView.findViewById(R.id.ivType); ivAvatar = itemView.findViewById(R.id.ivAvatar); tvName = itemView.findViewById(R.id.tvName); tvEmail = itemView.findViewById(R.id.tvEmail); tvTimes = itemView.findViewById(R.id.tvTimes); tvLast = itemView.findViewById(R.id.tvLast); ivFavorite = itemView.findViewById(R.id.ivFavorite); } Rect getItemRect() { return new Rect( super.itemView.getLeft(), super.itemView.getTop(), super.itemView.getRight(), super.itemView.getBottom()); } private void wire() { view.setOnClickListener(this); view.setOnLongClickListener(this); } private void unwire() { view.setOnClickListener(null); view.setOnLongClickListener(null); } private void bindTo(TupleContactEx contact) { view.setAlpha(contact.state == EntityContact.STATE_IGNORE ? Helper.LOW_LIGHT : 1.0f); if (contact.type == EntityContact.TYPE_FROM) { ivType.setImageResource(R.drawable.twotone_call_received_24); ivType.setContentDescription(context.getString(R.string.title_accessibility_from)); } else if (contact.type == EntityContact.TYPE_TO) { ivType.setImageResource(R.drawable.twotone_call_made_24); ivType.setContentDescription(context.getString(R.string.title_accessibility_to)); } else if (contact.type == EntityContact.TYPE_JUNK) { ivType.setImageResource(R.drawable.twotone_report_24); ivType.setContentDescription(context.getString(R.string.title_legend_junk)); } else if (contact.type == EntityContact.TYPE_NO_JUNK) { ivType.setImageResource(R.drawable.twotone_report_off_24); ivType.setContentDescription(context.getString(R.string.title_no_junk)); } else { ivType.setImageDrawable(null); ivType.setContentDescription(null); } if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) ivType.setTooltipText(ivType.getContentDescription()); if (contact.avatar == null || !contacts) ivAvatar.setImageDrawable(null); else { ContentResolver resolver = context.getContentResolver(); Uri lookupUri = Uri.parse(contact.avatar); try (InputStream is = ContactsContract.Contacts.openContactPhotoInputStream( resolver, lookupUri, false)) { ivAvatar.setImageBitmap(BitmapFactory.decodeStream(is)); } catch (Throwable ex) { Log.e(ex); } } tvName.setText(contact.name == null ? "-" : contact.name + (contact.group == null ? "" : "/" + contact.group)); tvEmail.setText(contact.email + "/" + contact.accountName); tvTimes.setText(NF.format(contact.times_contacted)); tvLast.setText(contact.last_contacted == null ? null : Helper.getRelativeTimeSpanString(context, contact.last_contacted)); ivFavorite.setImageResource(contact.state == EntityContact.STATE_FAVORITE ? R.drawable.baseline_star_24 : R.drawable.twotone_star_border_24); ivFavorite.setImageTintList(ColorStateList.valueOf( contact.state == EntityContact.STATE_FAVORITE ? colorAccent : textColorSecondary)); ivFavorite.setContentDescription(contact.state == EntityContact.STATE_FAVORITE ? context.getString(R.string.title_accessibility_flagged) : null); ivFavorite.setVisibility(contact.type == EntityContact.TYPE_JUNK || contact.type == EntityContact.TYPE_NO_JUNK ? View.GONE : View.VISIBLE); view.requestLayout(); } @Override public void onClick(View view) { int pos = getAdapterPosition(); if (pos == RecyclerView.NO_POSITION) return; TupleContactEx contact = selected.get(pos); if (contact.state == EntityContact.STATE_DEFAULT) contact.state = EntityContact.STATE_FAVORITE; else if (contact.state == EntityContact.STATE_FAVORITE) contact.state = EntityContact.STATE_IGNORE; else contact.state = EntityContact.STATE_DEFAULT; notifyItemChanged(pos); Bundle args = new Bundle(); args.putLong("id", contact.id); args.putInt("state", contact.state); new SimpleTask() { @Override protected Void onExecute(Context context, Bundle args) { long id = args.getLong("id"); int state = args.getInt("state"); DB db = DB.getInstance(context); db.contact().setContactState(id, state); return null; } @Override protected void onExecuted(Bundle args, Void data) { Shortcuts.update(context, owner); } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(parentFragment.getParentFragmentManager(), ex); } }.execute(context, owner, args, "contact:state"); } @Override public boolean onLongClick(View view) { int pos = getAdapterPosition(); if (pos == RecyclerView.NO_POSITION) return false; final TupleContactEx contact = selected.get(pos); final Intent share = new Intent(Intent.ACTION_INSERT); share.setType(ContactsContract.Contacts.CONTENT_TYPE); share.putExtra(ContactsContract.Intents.Insert.NAME, contact.name); share.putExtra(ContactsContract.Intents.Insert.EMAIL, contact.email); PopupMenuLifecycle popupMenu = new PopupMenuLifecycle(context, powner, view); int order = 0; SpannableString ss = new SpannableString(contact.email); ss.setSpan(new StyleSpan(Typeface.ITALIC), 0, ss.length(), 0); ss.setSpan(new RelativeSizeSpan(0.9f), 0, ss.length(), 0); popupMenu.getMenu().add(Menu.NONE, 0, order++, ss).setEnabled(false); if (!TextUtils.isEmpty(contact.identityEmail)) { String via = context.getString(R.string.title_via, contact.identityEmail); popupMenu.getMenu().add(Menu.NONE, 0, order++, via).setEnabled(false); } if (contact.state != EntityContact.STATE_IGNORE && contact.type != EntityContact.TYPE_JUNK && contact.type != EntityContact.TYPE_NO_JUNK) popupMenu.getMenu().add(Menu.NONE, R.string.title_advanced_never_favorite, order++, R.string.title_advanced_never_favorite); popupMenu.getMenu().add(Menu.NONE, R.string.title_share, order++, R.string.title_share); // should be system whitelisted popupMenu.getMenu().add(Menu.NONE, R.string.title_clipboard_copy, order++, R.string.title_clipboard_copy); // should be system whitelisted if (Shortcuts.can(context)) popupMenu.getMenu().add(Menu.NONE, R.string.title_pin, order++, R.string.title_pin); popupMenu.getMenu().add(Menu.NONE, R.string.title_edit_contact, order++, R.string.title_edit_contact); popupMenu.getMenu().add(Menu.NONE, R.string.title_search, order++, R.string.title_search); popupMenu.getMenu().add(Menu.NONE, R.string.title_delete, order++, R.string.title_delete); popupMenu.setOnMenuItemClickListener(new PopupMenu.OnMenuItemClickListener() { @Override public boolean onMenuItemClick(MenuItem item) { int itemId = item.getItemId(); if (itemId == R.string.title_advanced_never_favorite) { onActionNeverFavorite(); return true; } else if (itemId == R.string.title_share) { onActionShare(); return true; } else if (itemId == R.string.title_clipboard_copy) { onActionCopy(); return true; } else if (itemId == R.string.title_pin) { onActionPin(); return true; } else if (itemId == R.string.title_edit_contact) { onActionEdit(); return true; } else if (itemId == R.string.title_search) { onActionSearch(); return true; } else if (itemId == R.string.title_delete) { onActionDelete(); return true; } return false; } private void onActionNeverFavorite() { Bundle args = new Bundle(); args.putLong("id", contact.id); new SimpleTask() { @Override protected Void onExecute(Context context, Bundle args) { long id = args.getLong("id"); DB db = DB.getInstance(context); db.contact().setContactState(id, EntityContact.STATE_IGNORE); return null; } @Override protected void onExecuted(Bundle args, Void data) { Shortcuts.update(context, owner); } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(parentFragment.getParentFragmentManager(), ex); } }.execute(context, owner, args, "contact:favorite"); } private void onActionShare() { try { context.startActivity(share); } catch (Throwable ex) { Helper.reportNoViewer(context, share, ex); } } private void onActionCopy() { ClipboardManager clipboard = Helper.getSystemService(context, ClipboardManager.class); if (clipboard == null) return; String copy; try { InternetAddress address = new InternetAddress(contact.email, contact.name, StandardCharsets.UTF_8.name()); copy = MessageHelper.formatAddresses(new Address[]{address}); } catch (UnsupportedEncodingException ex) { Log.e(ex); copy = contact.email; } ClipData clip = ClipData.newPlainText(context.getString(R.string.app_name), copy); clipboard.setPrimaryClip(clip); if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) ToastEx.makeText(context, R.string.title_clipboard_copied, Toast.LENGTH_LONG).show(); } private void onActionPin() { ShortcutInfoCompat.Builder builder = Shortcuts.getShortcut(context, contact); Shortcuts.requestPinShortcut(context, builder.build()); } private void onActionEdit() { Bundle args = new Bundle(); args.putLong("id", contact.id); args.putLong("account", contact.account); args.putInt("type", contact.type); args.putString("email", contact.email); args.putString("name", contact.name); args.putString("group", contact.group); FragmentDialogContactEdit fragment = new FragmentDialogContactEdit(); fragment.setArguments(args); fragment.setTargetFragment(parentFragment, FragmentContacts.REQUEST_EDIT_CONTACT); fragment.show(parentFragment.getParentFragmentManager(), "contact:edit"); } private void onActionSearch() { Intent search = new Intent(context, ActivityView.class) .putExtra(Intent.EXTRA_PROCESS_TEXT, contact.email) .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK); context.startActivity(search); } private void onActionDelete() { Bundle args = new Bundle(); args.putLong("id", contact.id); new SimpleTask() { @Override protected Void onExecute(Context context, Bundle args) { long id = args.getLong("id"); DB db = DB.getInstance(context); db.contact().deleteContact(id); return null; } @Override protected void onExecuted(Bundle args, Void data) { Shortcuts.update(context, owner); } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(parentFragment.getParentFragmentManager(), ex); } }.execute(context, owner, args, "contact:delete"); } }); popupMenu.show(); return true; } } AdapterContact(Fragment parentFragment) { this.parentFragment = parentFragment; this.context = parentFragment.getContext(); this.owner = parentFragment.getViewLifecycleOwner(); this.inflater = LayoutInflater.from(context); this.contacts = Helper.hasPermission(context, Manifest.permission.READ_CONTACTS); this.colorAccent = Helper.resolveColor(context, androidx.appcompat.R.attr.colorAccent); this.textColorSecondary = Helper.resolveColor(context, android.R.attr.textColorSecondary); setHasStableIds(true); owner.getLifecycle().addObserver(new LifecycleObserver() { @OnLifecycleEvent(Lifecycle.Event.ON_DESTROY) public void onDestroyed() { Log.d(AdapterContact.this + " parent destroyed"); AdapterContact.this.parentFragment = null; owner.getLifecycle().removeObserver(this); } }); } public void set(@NonNull List contacts) { Log.i("Set contacts=" + contacts.size() + " search=" + search + " account=" + account + " types=" + types.size()); all = contacts; new SimpleTask>() { @Override protected List onExecute(Context context, Bundle args) { List filtered; if (account == null && types.size() == 0) filtered = contacts; else { filtered = new ArrayList<>(); for (TupleContactEx contact : contacts) if ((account == null || contact.account.equals(account)) && (types.size() == 0 || types.contains(contact.type))) filtered.add(contact); } List items; if (TextUtils.isEmpty(search)) items = filtered; else { items = new ArrayList<>(); String query = search.toLowerCase().trim(); for (TupleContactEx contact : filtered) if (contact.email.toLowerCase().contains(query) || (contact.name != null && contact.name.toLowerCase().contains(query))) items.add(contact); } return items; } @Override protected void onExecuted(Bundle args, List items) { DiffUtil.DiffResult diff = DiffUtil.calculateDiff(new DiffCallback(selected, items), false); selected = items; diff.dispatchUpdatesTo(new ListUpdateCallback() { @Override public void onInserted(int position, int count) { Log.d("Inserted @" + position + " #" + count); } @Override public void onRemoved(int position, int count) { Log.d("Removed @" + position + " #" + count); } @Override public void onMoved(int fromPosition, int toPosition) { Log.d("Moved " + fromPosition + ">" + toPosition); } @Override public void onChanged(int position, int count, Object payload) { Log.d("Changed @" + position + " #" + count); } }); try { diff.dispatchUpdatesTo(AdapterContact.this); } catch (Throwable ex) { Log.e(ex); } } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(parentFragment.getParentFragmentManager(), ex); } }.serial().execute(context, owner, new Bundle(), "contacts:filter"); } public void search(String query) { Log.i("Contacts query=" + query); search = query; set(all); } public void filter(Long account, boolean junk) { this.account = account; this.types = junk ? Arrays.asList(EntityContact.TYPE_JUNK, EntityContact.TYPE_NO_JUNK) : Arrays.asList(EntityContact.TYPE_TO, EntityContact.TYPE_FROM); set(all); } private static class DiffCallback extends DiffUtil.Callback { private List prev = new ArrayList<>(); private List next = new ArrayList<>(); DiffCallback(List prev, List next) { this.prev.addAll(prev); this.next.addAll(next); } @Override public int getOldListSize() { return prev.size(); } @Override public int getNewListSize() { return next.size(); } @Override public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) { TupleContactEx c1 = prev.get(oldItemPosition); TupleContactEx c2 = next.get(newItemPosition); return c1.id.equals(c2.id); } @Override public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) { TupleContactEx c1 = prev.get(oldItemPosition); TupleContactEx c2 = next.get(newItemPosition); return c1.equals(c2); } } @Override public long getItemId(int position) { if (position < 0 || position >= selected.size()) return -1L; return selected.get(position).id; } @Override public int getItemCount() { return selected.size(); } @Override @NonNull public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { return new ViewHolder(inflater.inflate(R.layout.item_contact, parent, false)); } @Override public void onBindViewHolder(@NonNull ViewHolder holder, int position) { TupleContactEx contact = selected.get(position); holder.powner.recreate(contact == null ? null : contact.id); holder.unwire(); holder.bindTo(contact); holder.wire(); } }