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. NetGuard 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 NetGuard. If not, see . Copyright 2018 by Marcel Bokhorst (M66B) */ import android.content.Context; import android.content.Intent; import android.content.SharedPreferences; import android.graphics.Canvas; import android.graphics.drawable.Drawable; import android.os.Bundle; import android.os.Handler; import android.preference.PreferenceManager; import android.text.TextUtils; import android.util.Log; 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.ImageButton; import android.widget.ProgressBar; import android.widget.TextView; import com.google.android.material.floatingactionbutton.FloatingActionButton; import com.google.android.material.snackbar.Snackbar; import java.util.Date; import java.util.List; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.appcompat.widget.SearchView; import androidx.constraintlayout.widget.Group; import androidx.fragment.app.FragmentTransaction; import androidx.lifecycle.LiveData; import androidx.lifecycle.Observer; import androidx.paging.LivePagedListBuilder; import androidx.paging.PagedList; import androidx.recyclerview.widget.ItemTouchHelper; import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; public class FragmentMessages extends FragmentEx { private ViewGroup view; private TextView tvSupport; private ImageButton ibHintSupport; private ImageButton ibHintActions; private RecyclerView rvMessage; private TextView tvNoEmail; private ProgressBar pbWait; private Group grpSupport; private Group grpHintSupport; private Group grpHintActions; private Group grpReady; private FloatingActionButton fab; private long folder = -1; private long thread = -1; private String search = null; private long primary = -1; private AdapterMessage adapter; private AdapterMessage.ViewType viewType; private LiveData> messages = null; private SearchState searchState = SearchState.Reset; private BoundaryCallbackMessages searchCallback = null; private ExecutorService executor = Executors.newCachedThreadPool(Helper.backgroundThreadFactory); private static final int MESSAGES_PAGE_SIZE = 50; private static final int SEARCH_PAGE_SIZE = 10; private static final int UNDO_TIMEOUT = 5000; // milliseconds private enum SearchState {Reset, Database, Boundary} @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // Get arguments Bundle args = getArguments(); if (args != null) { folder = args.getLong("folder", -1); thread = args.getLong("thread", -1); // message ID search = args.getString("search"); } } @Override @Nullable public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { view = (ViewGroup) inflater.inflate(R.layout.fragment_messages, container, false); setHasOptionsMenu(true); // Get controls tvSupport = view.findViewById(R.id.tvSupport); ibHintSupport = view.findViewById(R.id.ibHintSupport); ibHintActions = view.findViewById(R.id.ibHintActions); rvMessage = view.findViewById(R.id.rvFolder); tvNoEmail = view.findViewById(R.id.tvNoEmail); pbWait = view.findViewById(R.id.pbWait); grpSupport = view.findViewById(R.id.grpSupport); grpHintSupport = view.findViewById(R.id.grpHintSupport); grpHintActions = view.findViewById(R.id.grpHintActions); grpReady = view.findViewById(R.id.grpReady); fab = view.findViewById(R.id.fab); final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getContext()); // Wire controls tvSupport.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction(); fragmentTransaction.replace(R.id.content_frame, new FragmentPro()).addToBackStack("pro"); fragmentTransaction.commit(); } }); ibHintActions.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { prefs.edit().putBoolean("message_actions", true).apply(); grpHintActions.setVisibility(View.GONE); } }); ibHintSupport.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { prefs.edit().putBoolean("app_support", true).apply(); grpHintSupport.setVisibility(View.GONE); } }); rvMessage.setHasFixedSize(false); LinearLayoutManager llm = new LinearLayoutManager(getContext()); rvMessage.setLayoutManager(llm); if (TextUtils.isEmpty(search)) if (thread < 0) if (folder < 0) viewType = AdapterMessage.ViewType.UNIFIED; else viewType = AdapterMessage.ViewType.FOLDER; else viewType = AdapterMessage.ViewType.THREAD; else viewType = AdapterMessage.ViewType.SEARCH; adapter = new AdapterMessage(getContext(), getViewLifecycleOwner(), viewType); rvMessage.setAdapter(adapter); new ItemTouchHelper(new ItemTouchHelper.Callback() { @Override public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { int pos = viewHolder.getAdapterPosition(); if (pos == RecyclerView.NO_POSITION) return 0; TupleMessageEx message = ((AdapterMessage) rvMessage.getAdapter()).getCurrentList().get(pos); if (EntityFolder.OUTBOX.equals(message.folderType)) return 0; return makeMovementFlags(0, ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT); } @Override public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) { return false; } @Override public void onChildDraw(Canvas canvas, RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, float dX, float dY, int actionState, boolean isCurrentlyActive) { int pos = viewHolder.getAdapterPosition(); if (pos == RecyclerView.NO_POSITION) return; TupleMessageEx message = ((AdapterMessage) rvMessage.getAdapter()).getCurrentList().get(pos); boolean inbox = (EntityFolder.ARCHIVE.equals(message.folderType) || EntityFolder.TRASH.equals(message.folderType)); View itemView = viewHolder.itemView; int margin = Math.round(12 * (getResources().getDisplayMetrics().density)); if (dX > margin) { // Right swipe Drawable d = getResources().getDrawable(inbox ? R.drawable.baseline_inbox_24 : R.drawable.baseline_archive_24, getContext().getTheme()); int padding = (itemView.getHeight() - d.getIntrinsicHeight()); d.setBounds( itemView.getLeft() + margin, itemView.getTop() + padding / 2, itemView.getLeft() + margin + d.getIntrinsicWidth(), itemView.getTop() + padding / 2 + d.getIntrinsicHeight()); d.draw(canvas); } else if (dX < -margin) { // Left swipe Drawable d = getResources().getDrawable(inbox ? R.drawable.baseline_inbox_24 : R.drawable.baseline_delete_24, getContext().getTheme()); int padding = (itemView.getHeight() - d.getIntrinsicHeight()); d.setBounds( itemView.getLeft() + itemView.getWidth() - d.getIntrinsicWidth() - margin, itemView.getTop() + padding / 2, itemView.getLeft() + itemView.getWidth() - margin, itemView.getTop() + padding / 2 + d.getIntrinsicHeight()); d.draw(canvas); } super.onChildDraw(canvas, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive); } @Override public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) { int pos = viewHolder.getAdapterPosition(); if (pos == RecyclerView.NO_POSITION) return; TupleMessageEx message = ((AdapterMessage) rvMessage.getAdapter()).getCurrentList().get(pos); Log.i(Helper.TAG, "Swiped dir=" + direction + " message=" + message.id); Bundle args = new Bundle(); args.putLong("id", message.id); args.putInt("direction", direction); new SimpleTask() { @Override protected String onLoad(Context context, Bundle args) { long id = args.getLong("id"); int direction = args.getInt("direction"); EntityFolder target = null; // Get target folder and hide message DB db = DB.getInstance(context); try { db.beginTransaction(); EntityMessage message = db.message().getMessage(id); EntityFolder folder = db.folder().getFolder(message.folder); if (EntityFolder.ARCHIVE.equals(folder.type) || EntityFolder.TRASH.equals(folder.type)) target = db.folder().getFolderByType(message.account, EntityFolder.INBOX); else { if (direction == ItemTouchHelper.RIGHT) target = db.folder().getFolderByType(message.account, EntityFolder.ARCHIVE); if (direction == ItemTouchHelper.LEFT || target == null) target = db.folder().getFolderByType(message.account, EntityFolder.TRASH); } db.message().setMessageUiHide(message.id, true); db.setTransactionSuccessful(); } finally { db.endTransaction(); } Log.i(Helper.TAG, "Move id=" + id + " target=" + target); return target.name; } @Override protected void onLoaded(final Bundle args, final String target) { // Show undo snackbar final Snackbar snackbar = Snackbar.make( view, getString(R.string.title_moving, Helper.localizeFolderName(getContext(), target)), Snackbar.LENGTH_INDEFINITE); snackbar.setAction(R.string.title_undo, new View.OnClickListener() { @Override public void onClick(View v) { snackbar.dismiss(); // Show message again new SimpleTask() { @Override protected Void onLoad(Context context, Bundle args) { long id = args.getLong("id"); Log.i(Helper.TAG, "Undo move id=" + id); DB.getInstance(context).message().setMessageUiHide(id, false); return null; } @Override protected void onException(Bundle args, Throwable ex) { super.onException(args, ex); } }.load(FragmentMessages.this, args); } }); snackbar.show(); // Wait new Handler().postDelayed(new Runnable() { @Override public void run() { Log.i(Helper.TAG, "Move timeout shown=" + snackbar.isShown()); // Remove snackbar if (snackbar.isShown()) snackbar.dismiss(); final Context context = getContext(); args.putString("target", target); // Process move in a thread // - the fragment could be gone executor.submit(new Runnable() { @Override public void run() { try { long id = args.getLong("id"); String target = args.getString("target"); DB db = DB.getInstance(context); try { db.beginTransaction(); EntityMessage message = db.message().getMessage(id); if (message != null && message.ui_hide) { Log.i(Helper.TAG, "Moving id=" + id + " target=" + target); EntityFolder folder = db.folder().getFolderByName(message.account, target); EntityOperation.queue(db, message, EntityOperation.MOVE, folder.id); } db.setTransactionSuccessful(); } finally { db.endTransaction(); } EntityOperation.process(context); } catch (Throwable ex) { Log.e(Helper.TAG, ex + "\n" + Log.getStackTraceString(ex)); } } }); } }, UNDO_TIMEOUT); } @Override protected void onException(Bundle args, Throwable ex) { Helper.unexpectedError(getContext(), ex); } }.load(FragmentMessages.this, args); } }).attachToRecyclerView(rvMessage); fab.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { startActivity(new Intent(getContext(), ActivityCompose.class) .putExtra("action", "new") .putExtra("account", (Long) fab.getTag()) ); } }); // Initialize tvNoEmail.setVisibility(View.GONE); grpReady.setVisibility(View.GONE); pbWait.setVisibility(View.VISIBLE); fab.setVisibility(View.GONE); return view; } @Override public void onActivityCreated(@Nullable Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getContext()); grpHintSupport.setVisibility(prefs.getBoolean("app_support", false) ? View.GONE : View.VISIBLE); grpHintActions.setVisibility(prefs.getBoolean("message_actions", false) ? View.GONE : View.VISIBLE); final DB db = DB.getInstance(getContext()); // Primary account db.account().livePrimaryAccount().observe(getViewLifecycleOwner(), new Observer() { @Override public void onChanged(EntityAccount account) { primary = (account == null ? -1 : account.id); getActivity().invalidateOptionsMenu(); } }); // Folder switch (viewType) { case UNIFIED: db.folder().liveUnified().observe(getViewLifecycleOwner(), new Observer>() { @Override public void onChanged(List folders) { int unseen = 0; if (folders != null) for (TupleFolderEx folder : folders) unseen += folder.unseen; String name = getString(R.string.title_folder_unified); if (unseen > 0) setSubtitle(getString(R.string.title_folder_unseen, name, unseen)); else setSubtitle(name); } }); break; case FOLDER: db.folder().liveFolderEx(folder).observe(getViewLifecycleOwner(), new Observer() { @Override public void onChanged(@Nullable TupleFolderEx folder) { if (folder == null) setSubtitle(null); else { String name = Helper.localizeFolderName(getContext(), folder.name); if (folder.unseen > 0) setSubtitle(getString(R.string.title_folder_unseen, name, folder.unseen)); else setSubtitle(name); } } }); break; case THREAD: setSubtitle(R.string.title_folder_thread); break; case SEARCH: setSubtitle(getString(R.string.title_searching, search)); break; } // Messages loadMessages(); // Compose FAB Bundle args = new Bundle(); args.putLong("folder", folder); args.putLong("thread", thread); new SimpleTask() { @Override protected Long onLoad(Context context, Bundle args) { long fid = args.getLong("folder", -1); long thread = args.getLong("thread", -1); // message ID DB db = DB.getInstance(context); Long account = null; if (thread < 0) { if (folder >= 0) { EntityFolder folder = db.folder().getFolder(fid); if (folder != null) account = folder.account; } } else { EntityMessage threaded = db.message().getMessage(thread); if (threaded != null) account = threaded.account; } if (account == null) { // outbox EntityFolder primary = db.folder().getPrimaryDrafts(); if (primary != null) account = primary.account; } return account; } @Override protected void onLoaded(Bundle args, Long account) { if (account != null) { fab.setTag(account); fab.setVisibility(View.VISIBLE); } } @Override protected void onException(Bundle args, Throwable ex) { Helper.unexpectedError(getContext(), ex); } }.load(this, args); } @Override public void onResume() { super.onResume(); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getContext()); grpSupport.setVisibility(prefs.getBoolean("pro", false) ? View.GONE : View.VISIBLE); } @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { inflater.inflate(R.menu.menu_list, menu); final MenuItem menuSearch = menu.findItem(R.id.menu_search); final SearchView searchView = (SearchView) menuSearch.getActionView(); searchView.setQueryHint(getString(R.string.title_search_hint)); searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() { @Override public boolean onQueryTextSubmit(String query) { menuSearch.collapseActionView(); if (PreferenceManager.getDefaultSharedPreferences(getContext()).getBoolean("pro", false)) { Intent intent = new Intent(); intent.putExtra("folder", folder); intent.putExtra("search", query); FragmentMessages fragment = new FragmentMessages(); fragment.setArguments(intent.getExtras()); FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction(); fragmentTransaction.replace(R.id.content_frame, fragment).addToBackStack("search"); fragmentTransaction.commit(); } else { FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction(); fragmentTransaction.replace(R.id.content_frame, new FragmentPro()).addToBackStack("pro"); fragmentTransaction.commit(); } return true; } @Override public boolean onQueryTextChange(String newText) { return false; } }); super.onCreateOptionsMenu(menu, inflater); } @Override public void onPrepareOptionsMenu(Menu menu) { menu.findItem(R.id.menu_search).setVisible(folder >= 0 && search == null); menu.findItem(R.id.menu_sort_on).setVisible(TextUtils.isEmpty(search)); menu.findItem(R.id.menu_folders).setVisible(primary >= 0); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getContext()); String sort = prefs.getString("sort", "time"); if ("time".equals(sort)) menu.findItem(R.id.menu_sort_on_time).setChecked(true); else if ("unread".equals(sort)) menu.findItem(R.id.menu_sort_on_unread).setChecked(true); else if ("starred".equals(sort)) menu.findItem(R.id.menu_sort_on_starred).setChecked(true); super.onPrepareOptionsMenu(menu); } @Override public boolean onOptionsItemSelected(MenuItem item) { SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getContext()); switch (item.getItemId()) { case R.id.menu_sort_on_time: prefs.edit().putString("sort", "time").apply(); item.setChecked(true); loadMessages(); return true; case R.id.menu_sort_on_unread: case R.id.menu_sort_on_starred: if (prefs.getBoolean("pro", false)) { prefs.edit().putString("sort", item.getItemId() == R.id.menu_sort_on_unread ? "unread" : "starred").apply(); item.setChecked(true); loadMessages(); } else { FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction(); fragmentTransaction.replace(R.id.content_frame, new FragmentPro()).addToBackStack("pro"); fragmentTransaction.commit(); } return true; case R.id.menu_folders: onMenuFolders(); loadMessages(); return true; default: return super.onOptionsItemSelected(item); } } private void onMenuFolders() { getFragmentManager().popBackStack("unified", 0); Bundle args = new Bundle(); args.putLong("account", primary); FragmentFolders fragment = new FragmentFolders(); fragment.setArguments(args); FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction(); fragmentTransaction.replace(R.id.content_frame, fragment).addToBackStack("folders"); fragmentTransaction.commit(); } private void loadMessages() { final DB db = DB.getInstance(getContext()); // Observe folder/messages/search if (TextUtils.isEmpty(search)) { SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getContext()); String sort = prefs.getString("sort", "time"); boolean debug = prefs.getBoolean("debug", false); if (messages != null) messages.removeObservers(getViewLifecycleOwner()); switch (viewType) { case UNIFIED: messages = new LivePagedListBuilder<>(db.message().pagedUnifiedInbox(sort, debug), MESSAGES_PAGE_SIZE).build(); break; case FOLDER: messages = new LivePagedListBuilder<>(db.message().pagedFolder(folder, sort, false, debug), MESSAGES_PAGE_SIZE).build(); break; case THREAD: messages = new LivePagedListBuilder<>(db.message().pagedThread(thread, sort, debug), MESSAGES_PAGE_SIZE).build(); break; } messages.observe(getViewLifecycleOwner(), new Observer>() { @Override public void onChanged(@Nullable PagedList messages) { if (messages == null) { finish(); return; } Log.i(Helper.TAG, "Submit messages=" + messages.size()); adapter.submitList(messages); pbWait.setVisibility(View.GONE); grpReady.setVisibility(View.VISIBLE); if (messages.size() == 0) { tvNoEmail.setVisibility(View.VISIBLE); rvMessage.setVisibility(View.GONE); } else { tvNoEmail.setVisibility(View.GONE); rvMessage.setVisibility(View.VISIBLE); } } }); } else { Log.i(Helper.TAG, "Search state=" + searchState); if (searchCallback == null) searchCallback = new BoundaryCallbackMessages( getContext(), FragmentMessages.this, folder, search, new BoundaryCallbackMessages.IBoundaryCallbackMessages() { @Override public void onLoading() { pbWait.setVisibility(View.VISIBLE); } @Override public void onLoaded() { pbWait.setVisibility(View.GONE); } @Override public void onError(Context context, Throwable ex) { Helper.unexpectedError(context, ex); } }); Bundle args = new Bundle(); args.putLong("folder", folder); args.putString("search", search); new SimpleTask() { @Override protected Void onLoad(Context context, Bundle args) { if (searchState == SearchState.Reset) { long folder = args.getLong("folder"); DB.getInstance(context).message().resetFound(folder); searchState = SearchState.Database; Log.i(Helper.TAG, "Search reset done"); } return null; } @Override protected void onLoaded(final Bundle args, Void data) { LivePagedListBuilder builder = new LivePagedListBuilder<>(db.message().pagedFolder(folder, "time", true, false), SEARCH_PAGE_SIZE); builder.setBoundaryCallback(searchCallback); LiveData> messages = builder.build(); messages.observe(getViewLifecycleOwner(), new Observer>() { @Override public void onChanged(PagedList messages) { Log.i(Helper.TAG, "Submit found messages=" + messages.size()); adapter.submitList(messages); grpReady.setVisibility(View.VISIBLE); } }); new SimpleTask() { @Override protected Long onLoad(Context context, Bundle args) throws Throwable { long last = 0; if (searchState == SearchState.Database) { last = new Date().getTime(); long folder = args.getLong("folder"); String search = args.getString("search").toLowerCase(); DB db = DB.getInstance(context); for (long id : db.message().getMessageIDs(folder)) { EntityMessage message = db.message().getMessage(id); if (message != null) { // Message could be removed in the meantime String from = MessageHelper.getFormattedAddresses(message.from, true); if (from.toLowerCase().contains(search) || message.subject.toLowerCase().contains(search) || message.read(context).toLowerCase().contains(search)) { Log.i(Helper.TAG, "Search found id=" + id); db.message().setMessageFound(message.id, true); last = message.received; } } } searchState = SearchState.Boundary; Log.i(Helper.TAG, "Search database done"); } return last; } @Override protected void onLoaded(Bundle args, Long last) { pbWait.setVisibility(View.GONE); searchCallback.setEnabled(true); if (last > 0) searchCallback.load(last); } }.load(FragmentMessages.this, args); } }.load(this, args); } } void onNewMessages() { rvMessage.scrollToPosition(0); } }