|
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 <http://www.gnu.org/licenses/>.
|
|
|
|
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<PagedList<TupleMessageEx>> 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<String>() {
|
|
@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<Void>() {
|
|
@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<EntityAccount>() {
|
|
@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<List<TupleFolderEx>>() {
|
|
@Override
|
|
public void onChanged(List<TupleFolderEx> 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<TupleFolderEx>() {
|
|
@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<Long>() {
|
|
@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<PagedList<TupleMessageEx>>() {
|
|
@Override
|
|
public void onChanged(@Nullable PagedList<TupleMessageEx> 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<Void>() {
|
|
@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<Integer, TupleMessageEx> builder = new LivePagedListBuilder<>(db.message().pagedFolder(folder, "time", true, false), SEARCH_PAGE_SIZE);
|
|
builder.setBoundaryCallback(searchCallback);
|
|
LiveData<PagedList<TupleMessageEx>> messages = builder.build();
|
|
messages.observe(getViewLifecycleOwner(), new Observer<PagedList<TupleMessageEx>>() {
|
|
@Override
|
|
public void onChanged(PagedList<TupleMessageEx> messages) {
|
|
Log.i(Helper.TAG, "Submit found messages=" + messages.size());
|
|
adapter.submitList(messages);
|
|
grpReady.setVisibility(View.VISIBLE);
|
|
}
|
|
});
|
|
|
|
new SimpleTask<Long>() {
|
|
@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);
|
|
}
|
|
}
|