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-2022 by Marcel Bokhorst (M66B) */ import static android.app.Activity.RESULT_OK; import android.Manifest; import android.app.Dialog; import android.content.ContentResolver; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.content.pm.PackageManager; import android.net.Uri; import android.os.Bundle; 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.widget.Toast; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.appcompat.app.AlertDialog; import androidx.appcompat.widget.SearchView; import androidx.constraintlayout.widget.Group; import androidx.lifecycle.Observer; import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; import java.io.BufferedInputStream; import java.io.FileNotFoundException; import java.io.InputStream; import java.io.OutputStream; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.Arrays; import java.util.Date; import java.util.List; import javax.mail.Address; import javax.mail.internet.InternetAddress; import ezvcard.VCard; import ezvcard.VCardVersion; import ezvcard.io.text.VCardReader; import ezvcard.io.text.VCardWriter; import ezvcard.property.Email; import ezvcard.property.FormattedName; public class FragmentContacts extends FragmentBase { private RecyclerView rvContacts; private ContentLoadingProgressBar pbWait; private Group grpReady; private long account; private boolean junk = false; private String searching = null; private AdapterContact adapter; private static final int REQUEST_ACCOUNT = 1; private static final int REQUEST_IMPORT = 2; private static final int REQUEST_EXPORT = 3; static final int REQUEST_EDIT_NAME = 4; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Bundle args = getArguments(); this.junk = (args != null && args.getBoolean("junk")); } @Override @Nullable public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { setSubtitle(junk ? R.string.title_block_sender : R.string.menu_contacts); setHasOptionsMenu(true); View view = inflater.inflate(R.layout.fragment_contacts, container, false); // Get controls rvContacts = view.findViewById(R.id.rvContacts); pbWait = view.findViewById(R.id.pbWait); grpReady = view.findViewById(R.id.grpReady); // Wire controls rvContacts.setHasFixedSize(true); LinearLayoutManager llm = new LinearLayoutManager(getContext()); rvContacts.setLayoutManager(llm); adapter = new AdapterContact(this); rvContacts.setAdapter(adapter); // Initialize grpReady.setVisibility(View.GONE); pbWait.setVisibility(View.VISIBLE); return view; } @Override public void onSaveInstanceState(Bundle outState) { outState.putLong("fair:account", account); outState.putBoolean("fair:junk", junk); outState.putString("fair:searching", searching); super.onSaveInstanceState(outState); } @Override public void onActivityCreated(@Nullable Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); if (savedInstanceState != null) { account = savedInstanceState.getLong("fair:account"); junk = savedInstanceState.getBoolean("fair:junk"); searching = savedInstanceState.getString("fair:searching"); } onMenuJunk(junk); adapter.search(searching); DB db = DB.getInstance(getContext()); db.contact().liveContacts().observe(getViewLifecycleOwner(), new Observer>() { @Override public void onChanged(List contacts) { if (contacts == null) contacts = new ArrayList<>(); adapter.set(contacts); pbWait.setVisibility(View.GONE); grpReady.setVisibility(View.VISIBLE); } }); Shortcuts.update(getContext(), getViewLifecycleOwner()); } @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { inflater.inflate(R.menu.menu_contacts, menu); MenuItem menuSearch = menu.findItem(R.id.menu_search); SearchView searchView = (SearchView) menuSearch.getActionView(); searchView.setQueryHint(getString(R.string.title_search)); if (!TextUtils.isEmpty(searching)) { menuSearch.expandActionView(); searchView.setQuery(searching, true); } searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() { @Override public boolean onQueryTextChange(String newText) { if (getView() != null) { searching = newText; adapter.search(newText); } return true; } @Override public boolean onQueryTextSubmit(String query) { searching = query; adapter.search(query); return true; } }); super.onCreateOptionsMenu(menu, inflater); } @Override public void onPrepareOptionsMenu(@NonNull Menu menu) { super.onPrepareOptionsMenu(menu); menu.findItem(R.id.menu_junk).setChecked(junk); } @Override public boolean onOptionsItemSelected(MenuItem item) { int itemId = item.getItemId(); if (itemId == R.id.menu_help) { onMenuHelp(); return true; } else if (itemId == R.id.menu_junk) { item.setChecked(!item.isChecked()); onMenuJunk(item.isChecked()); return true; } else if (itemId == R.id.menu_import) { onMenuVcard(false); return true; } else if (itemId == R.id.menu_export) { onMenuVcard(true); return true; } else if (itemId == R.id.menu_delete) { onMenuDelete(); return true; } return super.onOptionsItemSelected(item); } private void onMenuHelp() { Helper.viewFAQ(getContext(), 84); } private void onMenuJunk(boolean junk) { this.junk = junk; setSubtitle(junk ? R.string.title_block_sender : R.string.menu_contacts); adapter.filter(junk ? Arrays.asList(EntityContact.TYPE_JUNK, EntityContact.TYPE_NO_JUNK) : new ArrayList<>()); } private void onMenuVcard(boolean export) { Bundle args = new Bundle(); args.putBoolean("export", export); FragmentDialogSelectAccount fragment = new FragmentDialogSelectAccount(); fragment.setArguments(args); fragment.setTargetFragment(this, REQUEST_ACCOUNT); fragment.show(getParentFragmentManager(), "messages:accounts"); } private void onMenuDelete() { new FragmentDelete().show(getParentFragmentManager(), "contacts:delete"); } @Override public void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); try { switch (requestCode) { case REQUEST_ACCOUNT: if (resultCode == RESULT_OK && data != null) onAccountSelected(data.getBundleExtra("args")); break; case REQUEST_IMPORT: if (resultCode == RESULT_OK && data != null) handleImport(data); break; case REQUEST_EXPORT: if (resultCode == RESULT_OK && data != null) handleExport(data); break; case REQUEST_EDIT_NAME: if (resultCode == RESULT_OK && data != null) onEditName(data.getBundleExtra("args")); break; } } catch (Throwable ex) { Log.e(ex); } } private void onAccountSelected(Bundle args) { account = args.getLong("account"); boolean export = args.getBoolean("export"); final Context context = getContext(); PackageManager pm = context.getPackageManager(); if (export) { Intent intent = new Intent(Intent.ACTION_CREATE_DOCUMENT); intent.addCategory(Intent.CATEGORY_OPENABLE); intent.setType("*/*"); intent.putExtra(Intent.EXTRA_TITLE, "fairemail.vcf"); Helper.openAdvanced(intent); startActivityForResult(Helper.getChooser(context, intent), REQUEST_EXPORT); } else { Intent open = new Intent(Intent.ACTION_GET_CONTENT); open.addCategory(Intent.CATEGORY_OPENABLE); open.setType("*/*"); if (open.resolveActivity(pm) == null) // system whitelisted ToastEx.makeText(context, R.string.title_no_saf, Toast.LENGTH_LONG).show(); else startActivityForResult(Helper.getChooser(context, open), REQUEST_IMPORT); } } private void handleImport(Intent data) { Uri uri = data.getData(); Bundle args = new Bundle(); args.putParcelable("uri", uri); args.putLong("account", account); new SimpleTask() { @Override protected void onPreExecute(Bundle args) { ToastEx.makeText(getContext(), R.string.title_executing, Toast.LENGTH_LONG).show(); } @Override protected Void onExecute(Context context, Bundle args) throws Throwable { Uri uri = args.getParcelable("uri"); long account = args.getLong("account"); if (uri == null) throw new FileNotFoundException(); if (!"content".equals(uri.getScheme()) && !Helper.hasPermission(context, Manifest.permission.READ_EXTERNAL_STORAGE)) { Log.w("Import uri=" + uri); throw new IllegalArgumentException(context.getString(R.string.title_no_stream)); } long now = new Date().getTime(); Log.i("Reading URI=" + uri); ContentResolver resolver = context.getContentResolver(); try (InputStream is = new BufferedInputStream(resolver.openInputStream(uri))) { VCardReader reader = new VCardReader(is); VCard vcard; while ((vcard = reader.readNext()) != null) { List emails = vcard.getEmails(); if (emails == null) continue; FormattedName fn = vcard.getFormattedName(); String name = (fn == null) ? null : fn.getValue(); List
addresses = new ArrayList<>(); for (Email email : emails) { String address = email.getValue(); if (address == null) continue; addresses.add(new InternetAddress(address, name, StandardCharsets.UTF_8.name())); } EntityContact.update(context, account, addresses.toArray(new Address[0]), EntityContact.TYPE_TO, now); } } Log.i("Imported contacts"); return null; } @Override protected void onExecuted(Bundle args, Void data) { ToastEx.makeText(getContext(), R.string.title_completed, Toast.LENGTH_LONG).show(); } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(getParentFragmentManager(), ex); } }.execute(this, args, "setup:import"); } private void handleExport(Intent data) { Bundle args = new Bundle(); args.putParcelable("uri", data.getData()); args.putLong("account", account); new SimpleTask() { @Override protected void onPreExecute(Bundle args) { ToastEx.makeText(getContext(), R.string.title_executing, Toast.LENGTH_LONG).show(); } @Override protected Void onExecute(Context context, Bundle args) throws Throwable { Uri uri = args.getParcelable("uri"); if (uri == null) throw new FileNotFoundException(); String password = args.getString("password"); EntityLog.log(context, "Exporting " + uri); if (!"content".equals(uri.getScheme())) { Log.w("Export uri=" + uri); throw new IllegalArgumentException(context.getString(R.string.title_no_stream)); } List vcards = new ArrayList<>(); DB db = DB.getInstance(context); List contacts = db.contact().getContacts(account); for (EntityContact contact : contacts) if (contact.type == EntityContact.TYPE_TO || contact.type == EntityContact.TYPE_FROM) { VCard vcard = new VCard(); vcard.addEmail(contact.email); if (!TextUtils.isEmpty(contact.name)) vcard.setFormattedName(contact.name); vcards.add(vcard); } ContentResolver resolver = context.getContentResolver(); try (OutputStream os = resolver.openOutputStream(uri)) { VCardWriter writer = new VCardWriter(os, VCardVersion.V3_0); for (VCard vcard : vcards) writer.write(vcard); } Log.i("Exported data"); return null; } @Override protected void onExecuted(Bundle args, Void data) { ToastEx.makeText(getContext(), R.string.title_completed, Toast.LENGTH_LONG).show(); } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(getParentFragmentManager(), ex); } }.execute(this, args, ""); } private void onEditName(Bundle args) { new SimpleTask() { @Override protected Void onExecute(Context context, Bundle args) { long id = args.getLong("id"); String name = args.getString("name"); if (TextUtils.isEmpty(name)) name = null; DB db = DB.getInstance(context); db.contact().setContactName(id, name); return null; } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(getParentFragmentManager(), ex); } }.execute(this, args, "contact:name"); } public static class FragmentDelete extends FragmentDialogBase { @NonNull @Override public Dialog onCreateDialog(@Nullable Bundle savedInstanceState) { return new AlertDialog.Builder(getContext()) .setIcon(R.drawable.twotone_warning_24) .setTitle(getString(R.string.title_delete_contacts)) .setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { new SimpleTask() { @Override protected Void onExecute(Context context, Bundle args) { DB db = DB.getInstance(context); int count = db.contact().clearContacts(); Log.i("Cleared contacts=" + count); return null; } @Override protected void onException(Bundle args, Throwable ex) { Log.unexpectedError(getParentFragmentManager(), ex); } }.execute(getContext(), getActivity(), new Bundle(), "contacts:delete"); } }) .setNegativeButton(android.R.string.cancel, null) .create(); } } }