Improved POP support

pull/147/head
M66B 6 years ago
parent fb3a4bfb63
commit 3edfb7b0bf

@ -481,7 +481,10 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
// Selected / disabled // Selected / disabled
itemView.setActivated(selectionTracker != null && selectionTracker.isSelected(message.id)); itemView.setActivated(selectionTracker != null && selectionTracker.isSelected(message.id));
itemView.setAlpha( itemView.setAlpha(
message.uid == null && !EntityFolder.OUTBOX.equals(message.folderType) ? Helper.LOW_LIGHT : 1.0f); message.uid == null &&
!message.accountPop &&
!EntityFolder.OUTBOX.equals(message.folderType)
? Helper.LOW_LIGHT : 1.0f);
// Duplicate // Duplicate
if (viewType == ViewType.THREAD) { if (viewType == ViewType.THREAD) {
@ -842,9 +845,10 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
bnvActions.getMenu().findItem(R.id.action_delete).setVisible( bnvActions.getMenu().findItem(R.id.action_delete).setVisible(
(inTrash && message.msgid != null) || (inTrash && message.msgid != null) ||
message.accountPop ||
(!inTrash && hasTrash && message.uid != null) || (!inTrash && hasTrash && message.uid != null) ||
(inOutbox && (!TextUtils.isEmpty(message.error) || !message.identitySynchronize))); (inOutbox && (!TextUtils.isEmpty(message.error) || !message.identitySynchronize)));
bnvActions.getMenu().findItem(R.id.action_delete).setTitle(inTrash ? R.string.title_delete : R.string.title_trash); bnvActions.getMenu().findItem(R.id.action_delete).setTitle(inTrash || message.accountPop ? R.string.title_delete : R.string.title_trash);
bnvActions.getMenu().findItem(R.id.action_move).setVisible( bnvActions.getMenu().findItem(R.id.action_move).setVisible(
message.uid != null || (inOutbox && (message.ui_snoozed != null || message.error != null))); message.uid != null || (inOutbox && (message.ui_snoozed != null || message.error != null)));
@ -2161,7 +2165,7 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
popupMenu.getMenu().findItem(R.id.menu_reply_all).setEnabled(data.message.content); popupMenu.getMenu().findItem(R.id.menu_reply_all).setEnabled(data.message.content);
popupMenu.getMenu().findItem(R.id.menu_answer).setEnabled(data.message.content); popupMenu.getMenu().findItem(R.id.menu_answer).setEnabled(data.message.content);
popupMenu.getMenu().findItem(R.id.menu_unseen).setEnabled(data.message.uid != null); popupMenu.getMenu().findItem(R.id.menu_unseen).setEnabled(data.message.uid != null || data.message.accountPop);
popupMenu.getMenu().findItem(R.id.menu_junk).setEnabled(data.message.uid != null); popupMenu.getMenu().findItem(R.id.menu_junk).setEnabled(data.message.uid != null);
popupMenu.getMenu().findItem(R.id.menu_junk).setVisible( popupMenu.getMenu().findItem(R.id.menu_junk).setVisible(

@ -47,7 +47,7 @@ public interface DaoMessage {
" AND NOT folder.type = '" + EntityFolder.DRAFTS + "' THEN 0 ELSE 1 END)"; " AND NOT folder.type = '" + EntityFolder.DRAFTS + "' THEN 0 ELSE 1 END)";
@Query("SELECT message.*" + @Query("SELECT message.*" +
", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.notify AS accountNotify" + ", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.pop as accountPop, account.notify AS accountNotify" +
", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" + ", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" +
", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" + ", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" +
", COUNT(message.id) AS count" + ", COUNT(message.id) AS count" +
@ -87,7 +87,7 @@ public interface DaoMessage {
" AND NOT (folder.id <> :folder AND folder.type = '" + EntityFolder.DRAFTS + "') THEN 0 ELSE 1 END)"; " AND NOT (folder.id <> :folder AND folder.type = '" + EntityFolder.DRAFTS + "') THEN 0 ELSE 1 END)";
@Query("SELECT message.*" + @Query("SELECT message.*" +
", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.notify AS accountNotify" + ", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.pop as accountPop, account.notify AS accountNotify" +
", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" + ", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" +
", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" + ", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" +
", COUNT(message.id) AS count" + ", COUNT(message.id) AS count" +
@ -119,7 +119,7 @@ public interface DaoMessage {
DataSource.Factory<Integer, TupleMessageEx> pagedFolder(long folder, boolean threading, String sort, boolean snoozed, boolean found, boolean debug); DataSource.Factory<Integer, TupleMessageEx> pagedFolder(long folder, boolean threading, String sort, boolean snoozed, boolean found, boolean debug);
@Query("SELECT message.*" + @Query("SELECT message.*" +
", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.notify AS accountNotify" + ", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.pop as accountPop, account.notify AS accountNotify" +
", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" + ", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" +
", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" + ", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" +
", 1 AS count" + ", 1 AS count" +
@ -191,7 +191,7 @@ public interface DaoMessage {
int countMessageByMsgId(long folder, String msgid); int countMessageByMsgId(long folder, String msgid);
@Query("SELECT message.*" + @Query("SELECT message.*" +
", account.name AS accountName, identity.color AS accountColor, account.notify AS accountNotify" + ", account.name AS accountName, identity.color AS accountColor, account.pop as accountPop, account.notify AS accountNotify" +
", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" + ", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" +
", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" + ", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" +
", 1 AS count" + ", 1 AS count" +
@ -208,7 +208,7 @@ public interface DaoMessage {
LiveData<TupleMessageEx> liveMessage(long id); LiveData<TupleMessageEx> liveMessage(long id);
@Query("SELECT message.*" + @Query("SELECT message.*" +
", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.notify AS accountNotify" + ", account.name AS accountName, IFNULL(identity.color, account.color) AS accountColor, account.pop as accountPop, account.notify AS accountNotify" +
", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" + ", folder.name AS folderName, folder.display AS folderDisplay, folder.type AS folderType" +
", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" + ", identity.name AS identityName, identity.email AS identityEmail, identity.synchronize AS identitySynchronize" +
", 1 AS count" + ", 1 AS count" +

@ -635,11 +635,7 @@ public class FragmentAccount extends FragmentBase {
for (Folder ifolder : istore.getDefaultFolder().list("*")) { for (Folder ifolder : istore.getDefaultFolder().list("*")) {
// Check folder attributes // Check folder attributes
String fullName = ifolder.getFullName(); String fullName = ifolder.getFullName();
String[] attrs; String[] attrs = ((IMAPFolder) ifolder).getAttributes();
if (ifolder instanceof IMAPFolder)
attrs = ((IMAPFolder) ifolder).getAttributes();
else
attrs = new String[0];
Log.i(fullName + " attrs=" + TextUtils.join(" ", attrs)); Log.i(fullName + " attrs=" + TextUtils.join(" ", attrs));
String type = EntityFolder.getType(attrs, fullName); String type = EntityFolder.getType(attrs, fullName);
@ -862,6 +858,15 @@ public class FragmentAccount extends FragmentBase {
EntityFolder left = (EntityFolder) args.getSerializable("left"); EntityFolder left = (EntityFolder) args.getSerializable("left");
EntityFolder right = (EntityFolder) args.getSerializable("right"); EntityFolder right = (EntityFolder) args.getSerializable("right");
if (pop) {
drafts = new EntityFolder();
drafts.name = "Drafts";
drafts.synchronize = false;
drafts.initialize = false;
drafts.sync_days = 0;
drafts.keep_days = 0;
}
if (TextUtils.isEmpty(host)) if (TextUtils.isEmpty(host))
throw new IllegalArgumentException(context.getString(R.string.title_no_host)); throw new IllegalArgumentException(context.getString(R.string.title_no_host));
if (TextUtils.isEmpty(port)) if (TextUtils.isEmpty(port))
@ -947,8 +952,8 @@ public class FragmentAccount extends FragmentBase {
inbox.unified = true; inbox.unified = true;
inbox.notify = true; inbox.notify = true;
inbox.initialize = !pop; inbox.initialize = !pop;
inbox.sync_days = EntityFolder.DEFAULT_SYNC; inbox.sync_days = (pop ? 0 : EntityFolder.DEFAULT_SYNC);
inbox.keep_days = EntityFolder.DEFAULT_KEEP; inbox.keep_days = (pop ? 0 : EntityFolder.DEFAULT_KEEP);
} }
} }

@ -2171,7 +2171,11 @@ public class FragmentMessages extends FragmentBase {
db.beginTransaction(); db.beginTransaction();
EntityMessage message = db.message().getMessage(id); EntityMessage message = db.message().getMessage(id);
if (message.uid != null) { if (message == null)
return null;
EntityAccount account = db.account().getAccount(message.account);
if (message.uid != null || account.pop) {
if (!message.content) if (!message.content)
EntityOperation.queue(context, db, message, EntityOperation.BODY); EntityOperation.queue(context, db, message, EntityOperation.BODY);
if (!message.ui_seen) if (!message.ui_seen)

@ -951,7 +951,8 @@ public class ServiceSynchronize extends LifecycleService {
EntityLog.log(this, account.name + " connected"); EntityLog.log(this, account.name + " connected");
// Update folder list // Update folder list
synchronizeFolders(account, istore, state); if (istore instanceof IMAPStore)
synchronizeFolders(account, istore, state);
// Open synchronizing folders // Open synchronizing folders
final ExecutorService pollExecutor = Executors.newSingleThreadExecutor(Helper.backgroundThreadFactory); final ExecutorService pollExecutor = Executors.newSingleThreadExecutor(Helper.backgroundThreadFactory);
@ -1473,63 +1474,77 @@ public class ServiceSynchronize extends LifecycleService {
if (message != null) if (message != null)
db.message().setMessageError(message.id, null); db.message().setMessageError(message.id, null);
if (message != null && message.uid == null && if (account != null && account.pop) {
!(EntityOperation.ADD.equals(op.name) || if (EntityOperation.SEEN.equals(op.name))
EntityOperation.DELETE.equals(op.name) || doSeen(folder, (POP3Folder) ifolder, message, jargs, db);
EntityOperation.SEND.equals(op.name) ||
EntityOperation.SYNC.equals(op.name)))
throw new IllegalArgumentException(op.name + " without uid " + op.args);
// Operations should use database transaction when needed else if (EntityOperation.ADD.equals(op.name))
; // Do nothing
if (EntityOperation.SEEN.equals(op.name)) else if (EntityOperation.DELETE.equals(op.name))
doSeen(folder, (IMAPFolder) ifolder, message, jargs, db); doDelete(folder, (POP3Folder) ifolder, message, jargs, db);
else if (EntityOperation.FLAG.equals(op.name)) else if (EntityOperation.SYNC.equals(op.name))
doFlag(folder, (IMAPFolder) ifolder, message, jargs, db); synchronizeMessages(account, folder, (POP3Folder) ifolder, jargs, state);
else if (EntityOperation.ANSWERED.equals(op.name)) else
doAnswered(folder, (IMAPFolder) ifolder, message, jargs, db); throw new MessagingException("Unknown operation name=" + op.name);
} else {
if (message != null && message.uid == null &&
!(EntityOperation.ADD.equals(op.name) ||
EntityOperation.DELETE.equals(op.name) ||
EntityOperation.SEND.equals(op.name) ||
EntityOperation.SYNC.equals(op.name)))
throw new IllegalArgumentException(op.name + " without uid " + op.args);
// Operations should use database transaction when needed
else if (EntityOperation.KEYWORD.equals(op.name)) if (EntityOperation.SEEN.equals(op.name))
doKeyword(folder, (IMAPFolder) ifolder, message, jargs, db); doSeen(folder, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.ADD.equals(op.name)) else if (EntityOperation.FLAG.equals(op.name))
doAdd(folder, isession, (IMAPStore) istore, (IMAPFolder) ifolder, message, jargs, db); doFlag(folder, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.MOVE.equals(op.name)) else if (EntityOperation.ANSWERED.equals(op.name))
doMove(folder, isession, (IMAPStore) istore, (IMAPFolder) ifolder, message, jargs, db); doAnswered(folder, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.DELETE.equals(op.name)) else if (EntityOperation.KEYWORD.equals(op.name))
doDelete(folder, (IMAPFolder) ifolder, message, jargs, db); doKeyword(folder, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.SEND.equals(op.name)) else if (EntityOperation.ADD.equals(op.name))
doSend(message, db); doAdd(folder, isession, (IMAPStore) istore, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.HEADERS.equals(op.name)) else if (EntityOperation.MOVE.equals(op.name))
doHeaders(folder, (IMAPFolder) ifolder, message, db); doMove(folder, isession, (IMAPStore) istore, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.RAW.equals(op.name)) else if (EntityOperation.DELETE.equals(op.name))
doRaw(folder, (IMAPFolder) ifolder, message, jargs, db); doDelete(folder, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.BODY.equals(op.name)) else if (EntityOperation.SEND.equals(op.name))
doBody(folder, (IMAPFolder) ifolder, message, db); doSend(message, db);
else if (EntityOperation.ATTACHMENT.equals(op.name)) else if (EntityOperation.HEADERS.equals(op.name))
doAttachment(folder, op, (IMAPFolder) ifolder, message, jargs, db); doHeaders(folder, (IMAPFolder) ifolder, message, db);
else if (EntityOperation.SYNC.equals(op.name)) else if (EntityOperation.RAW.equals(op.name))
if (EntityFolder.OUTBOX.equals(folder.type)) doRaw(folder, (IMAPFolder) ifolder, message, jargs, db);
db.folder().setFolderError(folder.id, null);
else { else if (EntityOperation.BODY.equals(op.name))
if (ifolder instanceof IMAPFolder) doBody(folder, (IMAPFolder) ifolder, message, db);
else if (EntityOperation.ATTACHMENT.equals(op.name))
doAttachment(folder, op, (IMAPFolder) ifolder, message, jargs, db);
else if (EntityOperation.SYNC.equals(op.name))
if (EntityFolder.OUTBOX.equals(folder.type))
db.folder().setFolderError(folder.id, null);
else
synchronizeMessages(account, folder, (IMAPFolder) ifolder, jargs, state); synchronizeMessages(account, folder, (IMAPFolder) ifolder, jargs, state);
else if (ifolder instanceof POP3Folder)
synchronizeMessages(account, folder, (POP3Folder) ifolder, jargs, state);
}
else else
throw new MessagingException("Unknown operation name=" + op.name); throw new MessagingException("Unknown operation name=" + op.name);
}
// Operation succeeded // Operation succeeded
db.operation().deleteOperation(op.id); db.operation().deleteOperation(op.id);
@ -1598,6 +1613,15 @@ public class ServiceSynchronize extends LifecycleService {
} }
} }
private void doSeen(EntityFolder folder, POP3Folder ifolder, EntityMessage message, JSONArray jargs, DB db) throws MessagingException, JSONException {
boolean seen = jargs.getBoolean(0);
if (message.seen.equals(seen))
return;
Log.i("Setting POP message=" + message.id + " seen=" + seen);
db.message().setMessageSeen(message.id, seen);
}
private void doSeen(EntityFolder folder, IMAPFolder ifolder, EntityMessage message, JSONArray jargs, DB db) throws MessagingException, JSONException { private void doSeen(EntityFolder folder, IMAPFolder ifolder, EntityMessage message, JSONArray jargs, DB db) throws MessagingException, JSONException {
// Mark message (un)seen // Mark message (un)seen
if (!ifolder.getPermanentFlags().contains(Flags.Flag.SEEN)) { if (!ifolder.getPermanentFlags().contains(Flags.Flag.SEEN)) {
@ -1880,6 +1904,24 @@ public class ServiceSynchronize extends LifecycleService {
} }
} }
private void doDelete(EntityFolder folder, POP3Folder ifolder, EntityMessage message, JSONArray jargs, DB db) throws MessagingException {
Log.i("Deleting POP message=" + message.id + " msgid=" + message.msgid);
// Delete message
if (TextUtils.isEmpty(message.msgid))
throw new IllegalArgumentException("Message ID missing");
Message[] imessages = ifolder.search(new MessageIDTerm(message.msgid));
for (Message imessage : imessages) {
Log.i(folder.name + " deleting uid=" + message.uid + " msgid=" + message.msgid);
imessage.setFlag(Flags.Flag.DELETED, true);
}
ifolder.close();
ifolder.open(Folder.READ_WRITE);
db.message().deleteMessage(message.id);
}
private void doDelete(EntityFolder folder, IMAPFolder ifolder, EntityMessage message, JSONArray jargs, DB db) throws MessagingException { private void doDelete(EntityFolder folder, IMAPFolder ifolder, EntityMessage message, JSONArray jargs, DB db) throws MessagingException {
// Delete message // Delete message
if (TextUtils.isEmpty(message.msgid)) if (TextUtils.isEmpty(message.msgid))
@ -2234,11 +2276,7 @@ public class ServiceSynchronize extends LifecycleService {
for (Folder ifolder : ifolders) { for (Folder ifolder : ifolders) {
String fullName = ifolder.getFullName(); String fullName = ifolder.getFullName();
String[] attrs; String[] attrs = ((IMAPFolder) ifolder).getAttributes();
if (ifolder instanceof IMAPFolder)
attrs = ((IMAPFolder) ifolder).getAttributes();
else
attrs = new String[0];
String type = EntityFolder.getType(attrs, fullName); String type = EntityFolder.getType(attrs, fullName);
EntityLog.log(this, account.name + ":" + fullName + EntityLog.log(this, account.name + ":" + fullName +

@ -24,6 +24,7 @@ import androidx.room.Ignore;
public class TupleMessageEx extends EntityMessage { public class TupleMessageEx extends EntityMessage {
public String accountName; public String accountName;
public Integer accountColor; public Integer accountColor;
public boolean accountPop;
public boolean accountNotify; public boolean accountNotify;
public String folderName; public String folderName;
public String folderDisplay; public String folderDisplay;
@ -47,6 +48,7 @@ public class TupleMessageEx extends EntityMessage {
return (super.uiEquals(obj) && return (super.uiEquals(obj) &&
(this.accountName == null ? other.accountName == null : this.accountName.equals(other.accountName)) && (this.accountName == null ? other.accountName == null : this.accountName.equals(other.accountName)) &&
(this.accountColor == null ? other.accountColor == null : this.accountColor.equals(other.accountColor)) && (this.accountColor == null ? other.accountColor == null : this.accountColor.equals(other.accountColor)) &&
this.accountPop == other.accountPop &&
//this.accountNotify == other.accountNotify && //this.accountNotify == other.accountNotify &&
this.folderName.equals(other.folderName) && this.folderName.equals(other.folderName) &&
(this.folderDisplay == null ? other.folderDisplay == null : this.folderDisplay.equals(other.folderDisplay)) && (this.folderDisplay == null ? other.folderDisplay == null : this.folderDisplay.equals(other.folderDisplay)) &&
@ -72,6 +74,7 @@ public class TupleMessageEx extends EntityMessage {
return (super.equals(obj) && return (super.equals(obj) &&
(this.accountName == null ? other.accountName == null : this.accountName.equals(other.accountName)) && (this.accountName == null ? other.accountName == null : this.accountName.equals(other.accountName)) &&
(this.accountColor == null ? other.accountColor == null : this.accountColor.equals(other.accountColor)) && (this.accountColor == null ? other.accountColor == null : this.accountColor.equals(other.accountColor)) &&
this.accountPop == other.accountPop &&
this.accountNotify == other.accountNotify && this.accountNotify == other.accountNotify &&
this.folderName.equals(other.folderName) && this.folderName.equals(other.folderName) &&
(this.folderDisplay == null ? other.folderDisplay == null : this.folderDisplay.equals(other.folderDisplay)) && (this.folderDisplay == null ? other.folderDisplay == null : this.folderDisplay.equals(other.folderDisplay)) &&

Loading…
Cancel
Save