|
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.Manifest;
|
|
import android.content.ContentResolver;
|
|
import android.content.Context;
|
|
import android.content.DialogInterface;
|
|
import android.content.Intent;
|
|
import android.content.SharedPreferences;
|
|
import android.content.pm.PackageManager;
|
|
import android.graphics.Color;
|
|
import android.graphics.Typeface;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.net.Uri;
|
|
import android.os.Bundle;
|
|
import android.preference.PreferenceManager;
|
|
import android.provider.ContactsContract;
|
|
import android.text.format.DateUtils;
|
|
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.ProgressBar;
|
|
import android.widget.TextView;
|
|
|
|
import java.io.InputStream;
|
|
import java.text.DateFormat;
|
|
import java.text.SimpleDateFormat;
|
|
import java.util.Date;
|
|
import java.util.List;
|
|
import java.util.concurrent.ExecutorService;
|
|
import java.util.concurrent.Executors;
|
|
|
|
import androidx.annotation.NonNull;
|
|
import androidx.appcompat.widget.PopupMenu;
|
|
import androidx.core.content.ContextCompat;
|
|
import androidx.lifecycle.LifecycleOwner;
|
|
import androidx.lifecycle.Observer;
|
|
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
|
|
import androidx.paging.PagedListAdapter;
|
|
import androidx.recyclerview.widget.DiffUtil;
|
|
import androidx.recyclerview.widget.RecyclerView;
|
|
|
|
public class AdapterMessage extends PagedListAdapter<TupleMessageEx, AdapterMessage.ViewHolder> {
|
|
private Context context;
|
|
private LifecycleOwner owner;
|
|
private ViewType viewType;
|
|
|
|
private boolean avatars;
|
|
private boolean debug;
|
|
|
|
private ExecutorService executor = Executors.newCachedThreadPool(Helper.backgroundThreadFactory);
|
|
private DateFormat df = SimpleDateFormat.getDateTimeInstance(SimpleDateFormat.SHORT, SimpleDateFormat.LONG);
|
|
|
|
enum ViewType {UNIFIED, FOLDER, THREAD, SEARCH}
|
|
|
|
public class ViewHolder extends RecyclerView.ViewHolder
|
|
implements View.OnClickListener, View.OnLongClickListener {
|
|
View itemView;
|
|
View vwColor;
|
|
ImageView ivAvatar;
|
|
ImageView ivFlagged;
|
|
TextView tvFrom;
|
|
TextView tvSize;
|
|
TextView tvTime;
|
|
ImageView ivAttachments;
|
|
TextView tvSubject;
|
|
TextView tvFolder;
|
|
TextView tvCount;
|
|
ImageView ivThread;
|
|
TextView tvError;
|
|
ProgressBar pbLoading;
|
|
|
|
private static final int action_flag = 1;
|
|
private static final int action_seen = 2;
|
|
private static final int action_delete = 3;
|
|
|
|
ViewHolder(View itemView) {
|
|
super(itemView);
|
|
|
|
this.itemView = itemView.findViewById(R.id.clItem);
|
|
vwColor = itemView.findViewById(R.id.vwColor);
|
|
ivFlagged = itemView.findViewById(R.id.ivFlagged);
|
|
ivAvatar = itemView.findViewById(R.id.ivAvatar);
|
|
tvFrom = itemView.findViewById(R.id.tvFrom);
|
|
tvSize = itemView.findViewById(R.id.tvSize);
|
|
tvTime = itemView.findViewById(R.id.tvTime);
|
|
ivAttachments = itemView.findViewById(R.id.ivAttachments);
|
|
tvSubject = itemView.findViewById(R.id.tvSubject);
|
|
tvFolder = itemView.findViewById(R.id.tvFolder);
|
|
tvCount = itemView.findViewById(R.id.tvCount);
|
|
ivThread = itemView.findViewById(R.id.ivThread);
|
|
tvError = itemView.findViewById(R.id.tvError);
|
|
pbLoading = itemView.findViewById(R.id.pbLoading);
|
|
}
|
|
|
|
private void wire() {
|
|
itemView.setOnClickListener(this);
|
|
itemView.setOnLongClickListener(this);
|
|
}
|
|
|
|
private void unwire() {
|
|
itemView.setOnClickListener(null);
|
|
itemView.setOnLongClickListener(null);
|
|
}
|
|
|
|
private void clear() {
|
|
vwColor.setBackgroundColor(Color.TRANSPARENT);
|
|
ivFlagged.setVisibility(View.GONE);
|
|
ivAvatar.setVisibility(View.GONE);
|
|
tvFrom.setText(null);
|
|
tvSize.setText(null);
|
|
tvTime.setText(null);
|
|
ivAttachments.setVisibility(View.GONE);
|
|
tvSubject.setText(null);
|
|
tvFolder.setText(null);
|
|
tvCount.setText(null);
|
|
ivThread.setVisibility(View.GONE);
|
|
tvError.setVisibility(View.GONE);
|
|
pbLoading.setVisibility(View.VISIBLE);
|
|
}
|
|
|
|
private void bindTo(final TupleMessageEx message) {
|
|
pbLoading.setVisibility(View.GONE);
|
|
|
|
itemView.setAlpha(viewType == ViewType.THREAD && EntityFolder.ARCHIVE.equals(message.folderType) ? 0.5f : 1.0f);
|
|
|
|
boolean photo = false;
|
|
if (avatars && message.avatar != null) {
|
|
ContentResolver resolver = context.getContentResolver();
|
|
InputStream is = ContactsContract.Contacts.openContactPhotoInputStream(resolver, Uri.parse(message.avatar));
|
|
if (is != null) {
|
|
photo = true;
|
|
ivAvatar.setImageDrawable(Drawable.createFromStream(is, "avatar"));
|
|
}
|
|
}
|
|
ivAvatar.setVisibility(photo ? View.VISIBLE : View.GONE);
|
|
|
|
vwColor.setBackgroundColor(message.accountColor == null ? Color.TRANSPARENT : message.accountColor);
|
|
vwColor.setVisibility(viewType == ViewType.UNIFIED && message.accountColor != null ? View.VISIBLE : View.GONE);
|
|
|
|
if (viewType == ViewType.THREAD)
|
|
ivFlagged.setVisibility(message.unflagged == 1 ? View.GONE : View.VISIBLE);
|
|
else
|
|
ivFlagged.setVisibility(message.count - message.unflagged > 0 ? View.VISIBLE : View.GONE);
|
|
|
|
if (EntityFolder.DRAFTS.equals(message.folderType) ||
|
|
EntityFolder.OUTBOX.equals(message.folderType) ||
|
|
EntityFolder.SENT.equals(message.folderType)) {
|
|
tvFrom.setText(MessageHelper.getFormattedAddresses(message.to, false));
|
|
tvTime.setText(DateUtils.getRelativeTimeSpanString(context, message.sent == null ? message.received : message.sent));
|
|
} else {
|
|
tvFrom.setText(MessageHelper.getFormattedAddresses(message.from, false));
|
|
tvTime.setText(DateUtils.getRelativeTimeSpanString(context, message.received));
|
|
}
|
|
|
|
tvSize.setText(message.size == null ? null : Helper.humanReadableByteCount(message.size, true));
|
|
tvSize.setTypeface(null, message.content ? Typeface.NORMAL : Typeface.BOLD);
|
|
tvSize.setVisibility(message.size == null ? View.GONE : View.VISIBLE);
|
|
|
|
ivAttachments.setVisibility(message.attachments > 0 ? View.VISIBLE : View.GONE);
|
|
tvSubject.setText(message.subject);
|
|
|
|
if (viewType == ViewType.UNIFIED)
|
|
tvFolder.setText(message.accountName);
|
|
else if (viewType == ViewType.FOLDER)
|
|
tvFolder.setVisibility(View.GONE);
|
|
else {
|
|
String name = (message.folderDisplay == null
|
|
? Helper.localizeFolderName(context, message.folderName)
|
|
: message.folderDisplay);
|
|
tvFolder.setText(name);
|
|
}
|
|
|
|
if (viewType == ViewType.THREAD) {
|
|
tvCount.setVisibility(View.GONE);
|
|
ivThread.setVisibility(View.GONE);
|
|
} else {
|
|
tvCount.setText(Integer.toString(message.count));
|
|
ivThread.setVisibility(View.VISIBLE);
|
|
tvCount.setAlpha(message.threaded ? 1.0f : 0.5f);
|
|
ivThread.setAlpha(message.threaded ? 1.0f : 0.5f);
|
|
}
|
|
|
|
if (debug) {
|
|
DB db = DB.getInstance(context);
|
|
db.operation().getOperationsByMessage(message.id).removeObservers(owner);
|
|
db.operation().getOperationsByMessage(message.id).observe(owner, new Observer<List<EntityOperation>>() {
|
|
@Override
|
|
public void onChanged(List<EntityOperation> operations) {
|
|
String text = message.error +
|
|
"\n" + message.id + " " + df.format(new Date(message.received)) +
|
|
"\n" + (message.ui_hide ? "HIDDEN " : "") +
|
|
"seen=" + message.seen + "/" + message.ui_seen + "/" + message.unseen +
|
|
" " + message.uid + "/" + message.id +
|
|
"\n" + message.msgid;
|
|
if (operations != null)
|
|
for (EntityOperation op : operations)
|
|
text += "\n" + op.id + ":" + op.name + " " + df.format(new Date(op.created));
|
|
|
|
tvError.setText(text);
|
|
tvError.setVisibility(View.VISIBLE);
|
|
|
|
}
|
|
});
|
|
}
|
|
|
|
tvError.setText(message.error);
|
|
tvError.setVisibility(message.error == null ? View.GONE : View.VISIBLE);
|
|
|
|
int typeface = (message.unseen > 0 ? Typeface.BOLD : Typeface.NORMAL);
|
|
tvFrom.setTypeface(null, typeface);
|
|
tvTime.setTypeface(null, typeface);
|
|
tvSubject.setTypeface(null, typeface);
|
|
tvCount.setTypeface(null, typeface);
|
|
|
|
int colorUnseen = Helper.resolveColor(context, message.unseen > 0
|
|
? R.attr.colorUnread : android.R.attr.textColorSecondary);
|
|
tvFrom.setTextColor(colorUnseen);
|
|
tvTime.setTextColor(colorUnseen);
|
|
}
|
|
|
|
@Override
|
|
public void onClick(View view) {
|
|
int pos = getAdapterPosition();
|
|
if (pos == RecyclerView.NO_POSITION)
|
|
return;
|
|
TupleMessageEx message = getItem(pos);
|
|
|
|
if (EntityFolder.DRAFTS.equals(message.folderType))
|
|
context.startActivity(
|
|
new Intent(context, ActivityCompose.class)
|
|
.putExtra("action", "edit")
|
|
.putExtra("id", message.id));
|
|
else {
|
|
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(context);
|
|
lbm.sendBroadcast(
|
|
new Intent(ActivityView.ACTION_VIEW_MESSAGE)
|
|
.putExtra("message", message));
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean onLongClick(View view) {
|
|
int pos = getAdapterPosition();
|
|
if (pos == RecyclerView.NO_POSITION)
|
|
return false;
|
|
|
|
final TupleMessageEx message = getItem(pos);
|
|
|
|
PopupMenu popupMenu = new PopupMenu(context, itemView);
|
|
if (!message.threaded && !EntityFolder.OUTBOX.equals(message.folderType)) {
|
|
popupMenu.getMenu().add(Menu.NONE, action_flag, 1, message.ui_flagged ? R.string.title_unflag : R.string.title_flag);
|
|
popupMenu.getMenu().add(Menu.NONE, action_seen, 2, message.ui_seen ? R.string.title_unseen : R.string.title_seen);
|
|
}
|
|
if (EntityFolder.TRASH.equals(message.folderType))
|
|
popupMenu.getMenu().add(Menu.NONE, action_delete, 3, R.string.title_delete);
|
|
popupMenu.setOnMenuItemClickListener(new PopupMenu.OnMenuItemClickListener() {
|
|
@Override
|
|
public boolean onMenuItemClick(MenuItem target) {
|
|
Bundle args = new Bundle();
|
|
args.putLong("id", message.id);
|
|
args.putInt("action", target.getItemId());
|
|
|
|
if (target.getItemId() == action_delete) {
|
|
new DialogBuilderLifecycle(context, owner)
|
|
.setMessage(R.string.title_ask_delete)
|
|
.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
|
|
@Override
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
Bundle args = new Bundle();
|
|
args.putLong("id", message.id);
|
|
|
|
new SimpleTask<Void>() {
|
|
@Override
|
|
protected Void onLoad(Context context, Bundle args) {
|
|
long id = args.getLong("id");
|
|
|
|
DB db = DB.getInstance(context);
|
|
try {
|
|
db.beginTransaction();
|
|
|
|
EntityMessage message = db.message().getMessage(id);
|
|
db.message().setMessageUiHide(id, true);
|
|
EntityOperation.queue(db, message, EntityOperation.DELETE);
|
|
|
|
db.setTransactionSuccessful();
|
|
} finally {
|
|
db.endTransaction();
|
|
}
|
|
|
|
EntityOperation.process(context);
|
|
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
protected void onException(Bundle args, Throwable ex) {
|
|
Helper.unexpectedError(context, ex);
|
|
}
|
|
}.load(context, owner, args);
|
|
}
|
|
})
|
|
.setNegativeButton(android.R.string.cancel, null)
|
|
.show();
|
|
} else
|
|
new SimpleTask<Void>() {
|
|
@Override
|
|
protected Void onLoad(final Context context, Bundle args) {
|
|
long id = args.getLong("id");
|
|
int action = args.getInt("action");
|
|
|
|
DB db = DB.getInstance(context);
|
|
try {
|
|
db.beginTransaction();
|
|
|
|
EntityMessage message = db.message().getMessage(id);
|
|
if (action == action_flag) {
|
|
db.message().setMessageUiFlagged(message.id, !message.ui_flagged);
|
|
EntityOperation.queue(db, message, EntityOperation.FLAG, !message.ui_flagged);
|
|
} else if (action == action_seen) {
|
|
db.message().setMessageUiSeen(message.id, !message.ui_seen);
|
|
EntityOperation.queue(db, message, EntityOperation.SEEN, !message.ui_seen);
|
|
}
|
|
|
|
db.setTransactionSuccessful();
|
|
} finally {
|
|
db.endTransaction();
|
|
}
|
|
|
|
EntityOperation.process(context);
|
|
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public void onException(Bundle args, Throwable ex) {
|
|
Helper.unexpectedError(context, ex);
|
|
}
|
|
}.load(context, owner, args);
|
|
|
|
return true;
|
|
}
|
|
});
|
|
|
|
if (popupMenu.getMenu().hasVisibleItems())
|
|
popupMenu.show();
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
AdapterMessage(Context context, LifecycleOwner owner, ViewType viewType) {
|
|
super(DIFF_CALLBACK);
|
|
this.context = context;
|
|
this.owner = owner;
|
|
this.viewType = viewType;
|
|
|
|
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
|
|
|
this.avatars = (prefs.getBoolean("avatars", true) &&
|
|
ContextCompat.checkSelfPermission(context, Manifest.permission.READ_CONTACTS)
|
|
== PackageManager.PERMISSION_GRANTED);
|
|
this.debug = prefs.getBoolean("debug", false);
|
|
}
|
|
|
|
private static final DiffUtil.ItemCallback<TupleMessageEx> DIFF_CALLBACK =
|
|
new DiffUtil.ItemCallback<TupleMessageEx>() {
|
|
@Override
|
|
public boolean areItemsTheSame(
|
|
@NonNull TupleMessageEx prev, @NonNull TupleMessageEx next) {
|
|
return prev.id.equals(next.id);
|
|
}
|
|
|
|
@Override
|
|
public boolean areContentsTheSame(
|
|
@NonNull TupleMessageEx prev, @NonNull TupleMessageEx next) {
|
|
return prev.equals(next);
|
|
}
|
|
};
|
|
|
|
@Override
|
|
@NonNull
|
|
public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
|
|
return new ViewHolder(LayoutInflater.from(context).inflate(R.layout.item_message, parent, false));
|
|
}
|
|
|
|
@Override
|
|
public void onBindViewHolder(@NonNull ViewHolder holder, int position) {
|
|
holder.unwire();
|
|
|
|
TupleMessageEx message = getItem(position);
|
|
if (message == null)
|
|
holder.clear();
|
|
else {
|
|
holder.bindTo(message);
|
|
holder.wire();
|
|
}
|
|
}
|
|
}
|