mirror of
https://github.com/the-djmaze/snappymail.git
synced 2025-01-09 00:08:18 +08:00
1055 lines
30 KiB
JavaScript
1055 lines
30 KiB
JavaScript
import 'External/User/ko';
|
|
|
|
import { pInt, pString } from 'Common/Utils';
|
|
import { isPosNumeric, delegateRunOnDestroy, mailToHelper } from 'Common/UtilsUser';
|
|
|
|
import {
|
|
Capa,
|
|
StorageResultType,
|
|
Notification
|
|
} from 'Common/Enums';
|
|
|
|
import {
|
|
Layout,
|
|
FolderType,
|
|
SetSystemFoldersNotification,
|
|
MessageSetAction,
|
|
ClientSideKeyName
|
|
} from 'Common/EnumsUser';
|
|
|
|
import { doc, $htmlCL, Settings, leftPanelDisabled } from 'Common/Globals';
|
|
|
|
import { UNUSED_OPTION_VALUE } from 'Common/Consts';
|
|
|
|
import {
|
|
MessageFlagsCache,
|
|
setFolderHash,
|
|
getFolderHash,
|
|
getFolderInboxName,
|
|
getFolderFromCacheList
|
|
} from 'Common/Cache';
|
|
|
|
import {
|
|
userBackground,
|
|
mailBox,
|
|
root,
|
|
openPgpWorkerJs,
|
|
openPgpJs
|
|
} from 'Common/Links';
|
|
|
|
import { getNotification, i18n } from 'Common/Translator';
|
|
|
|
import AppStore from 'Stores/User/App';
|
|
import SettingsStore from 'Stores/User/Settings';
|
|
import NotificationStore from 'Stores/User/Notification';
|
|
import AccountStore from 'Stores/User/Account';
|
|
import ContactStore from 'Stores/User/Contact';
|
|
import IdentityStore from 'Stores/User/Identity';
|
|
import TemplateStore from 'Stores/User/Template';
|
|
import FolderStore from 'Stores/User/Folder';
|
|
import PgpStore from 'Stores/User/Pgp';
|
|
import MessageStore from 'Stores/User/Message';
|
|
import QuotaStore from 'Stores/User/Quota';
|
|
|
|
import * as Local from 'Storage/Client';
|
|
|
|
import Remote from 'Remote/User/Fetch';
|
|
|
|
import { EmailModel } from 'Model/Email';
|
|
import { AccountModel } from 'Model/Account';
|
|
import { IdentityModel } from 'Model/Identity';
|
|
import { TemplateModel } from 'Model/Template';
|
|
import { OpenPgpKeyModel } from 'Model/OpenPgpKey';
|
|
|
|
import { LoginUserScreen } from 'Screen/User/Login';
|
|
import { MailBoxUserScreen } from 'Screen/User/MailBox';
|
|
import { SettingsUserScreen } from 'Screen/User/Settings';
|
|
|
|
import { startScreens, showScreenPopup } from 'Knoin/Knoin';
|
|
|
|
import { AbstractApp } from 'App/Abstract';
|
|
|
|
import { ComposePopupView } from 'View/Popup/Compose';
|
|
import { FolderSystemPopupView } from 'View/Popup/FolderSystem';
|
|
import { AskPopupView } from 'View/Popup/Ask';
|
|
import { TwoFactorConfigurationPopupView } from 'View/Popup/TwoFactorConfiguration';
|
|
|
|
class AppUser extends AbstractApp {
|
|
constructor() {
|
|
super(Remote);
|
|
|
|
this.moveCache = {};
|
|
|
|
this.quotaDebounce = this.quota.debounce(30000);
|
|
this.moveOrDeleteResponseHelper = this.moveOrDeleteResponseHelper.bind(this);
|
|
|
|
this.messagesMoveTrigger = this.messagesMoveTrigger.debounce(500);
|
|
|
|
// wakeUp
|
|
const interval = 3600000; // 60m
|
|
var lastTime = Date.now();
|
|
setInterval(() => {
|
|
const currentTime = Date.now();
|
|
if (currentTime > (lastTime + interval + 1000)) {
|
|
if (rl.hash.check()) {
|
|
this.reload();
|
|
}
|
|
Remote.jsVersion((sResult, oData) => {
|
|
if (StorageResultType.Success === sResult && oData && !oData.Result) {
|
|
this.reload();
|
|
}
|
|
}, Settings.app('version'));
|
|
}
|
|
lastTime = currentTime;
|
|
}, interval);
|
|
|
|
if (rl.hash.check()) {
|
|
this.reload();
|
|
}
|
|
|
|
if (Settings.get('UserBackgroundHash')) {
|
|
setTimeout(() => {
|
|
const img = userBackground(Settings.get('UserBackgroundHash'));
|
|
if (img) {
|
|
$htmlCL.add('UserBackground');
|
|
doc.body.style.backgroundImage = "url("+img+")";
|
|
}
|
|
}, 1000);
|
|
}
|
|
}
|
|
|
|
remote() {
|
|
return Remote;
|
|
}
|
|
|
|
reload() {
|
|
(Settings.app('inIframe') ? parent : window).location.reload();
|
|
}
|
|
|
|
reloadFlagsCurrentMessageListAndMessageFromCache() {
|
|
MessageStore.messageList.forEach(message =>
|
|
MessageFlagsCache.initMessage(message)
|
|
);
|
|
MessageFlagsCache.initMessage(MessageStore.message());
|
|
}
|
|
|
|
/**
|
|
* @param {boolean=} bDropPagePosition = false
|
|
* @param {boolean=} bDropCurrenFolderCache = false
|
|
*/
|
|
reloadMessageList(bDropPagePosition = false, bDropCurrenFolderCache = false) {
|
|
let iOffset = (MessageStore.messageListPage() - 1) * SettingsStore.messagesPerPage();
|
|
|
|
if (bDropCurrenFolderCache) {
|
|
setFolderHash(FolderStore.currentFolderFullNameRaw(), '');
|
|
}
|
|
|
|
if (bDropPagePosition) {
|
|
MessageStore.messageListPage(1);
|
|
MessageStore.messageListPageBeforeThread(1);
|
|
iOffset = 0;
|
|
|
|
rl.route.setHash(
|
|
mailBox(
|
|
FolderStore.currentFolderFullNameHash(),
|
|
MessageStore.messageListPage(),
|
|
MessageStore.messageListSearch(),
|
|
MessageStore.messageListThreadUid()
|
|
),
|
|
true,
|
|
true
|
|
);
|
|
}
|
|
|
|
MessageStore.messageListLoading(true);
|
|
Remote.messageList(
|
|
(sResult, oData, bCached) => {
|
|
if (StorageResultType.Success === sResult && oData && oData.Result) {
|
|
MessageStore.messageListError('');
|
|
MessageStore.messageListLoading(false);
|
|
|
|
MessageStore.setMessageList(oData, bCached);
|
|
} else if (StorageResultType.Unload === sResult) {
|
|
MessageStore.messageListError('');
|
|
MessageStore.messageListLoading(false);
|
|
} else if (StorageResultType.Abort !== sResult) {
|
|
MessageStore.messageList([]);
|
|
MessageStore.messageListLoading(false);
|
|
MessageStore.messageListError(
|
|
getNotification((oData && oData.ErrorCode) || Notification.CantGetMessageList)
|
|
);
|
|
}
|
|
},
|
|
FolderStore.currentFolderFullNameRaw(),
|
|
iOffset,
|
|
SettingsStore.messagesPerPage(),
|
|
MessageStore.messageListSearch(),
|
|
MessageStore.messageListThreadUid()
|
|
);
|
|
}
|
|
|
|
recacheInboxMessageList() {
|
|
Remote.messageList(()=>{}, getFolderInboxName(), 0, SettingsStore.messagesPerPage(), '', '', true);
|
|
}
|
|
|
|
/**
|
|
* @param {Function} fResultFunc
|
|
* @returns {boolean}
|
|
*/
|
|
contactsSync(fResultFunc) {
|
|
const oContacts = ContactStore.contacts;
|
|
if (
|
|
oContacts.importing() ||
|
|
oContacts.syncing() ||
|
|
!ContactStore.enableContactsSync() ||
|
|
!ContactStore.allowContactsSync()
|
|
) {
|
|
return false;
|
|
}
|
|
|
|
oContacts.syncing(true);
|
|
|
|
Remote.contactsSync((sResult, oData) => {
|
|
oContacts.syncing(false);
|
|
|
|
if (fResultFunc) {
|
|
fResultFunc(sResult, oData);
|
|
}
|
|
});
|
|
|
|
return true;
|
|
}
|
|
|
|
messagesMoveTrigger() {
|
|
const sTrashFolder = FolderStore.trashFolder(),
|
|
sSpamFolder = FolderStore.spamFolder();
|
|
|
|
Object.values(this.moveCache).forEach(item => {
|
|
const isSpam = sSpamFolder === item.To,
|
|
isTrash = sTrashFolder === item.To,
|
|
isHam = !isSpam && sSpamFolder === item.From && getFolderInboxName() === item.To;
|
|
|
|
Remote.messagesMove(
|
|
this.moveOrDeleteResponseHelper,
|
|
item.From,
|
|
item.To,
|
|
item.Uid,
|
|
isSpam ? 'SPAM' : isHam ? 'HAM' : '',
|
|
isSpam || isTrash
|
|
);
|
|
});
|
|
|
|
this.moveCache = {};
|
|
}
|
|
|
|
messagesMoveHelper(fromFolderFullNameRaw, toFolderFullNameRaw, uidsForMove) {
|
|
const hash = '$$' + fromFolderFullNameRaw + '$$' + toFolderFullNameRaw + '$$';
|
|
if (!this.moveCache[hash]) {
|
|
this.moveCache[hash] = {
|
|
From: fromFolderFullNameRaw,
|
|
To: toFolderFullNameRaw,
|
|
Uid: []
|
|
};
|
|
}
|
|
|
|
this.moveCache[hash].Uid = this.moveCache[hash].Uid.concat(uidsForMove).unique();
|
|
this.messagesMoveTrigger();
|
|
}
|
|
|
|
messagesCopyHelper(sFromFolderFullNameRaw, sToFolderFullNameRaw, aUidForCopy) {
|
|
Remote.messagesCopy(this.moveOrDeleteResponseHelper, sFromFolderFullNameRaw, sToFolderFullNameRaw, aUidForCopy);
|
|
}
|
|
|
|
messagesDeleteHelper(sFromFolderFullNameRaw, aUidForRemove) {
|
|
Remote.messagesDelete(this.moveOrDeleteResponseHelper, sFromFolderFullNameRaw, aUidForRemove);
|
|
}
|
|
|
|
moveOrDeleteResponseHelper(sResult, oData) {
|
|
if (StorageResultType.Success === sResult && FolderStore.currentFolder()) {
|
|
if (oData && Array.isArray(oData.Result) && 2 === oData.Result.length) {
|
|
setFolderHash(oData.Result[0], oData.Result[1]);
|
|
} else {
|
|
setFolderHash(FolderStore.currentFolderFullNameRaw(), '');
|
|
|
|
if (oData && [Notification.CantMoveMessage, Notification.CantCopyMessage].includes(oData.ErrorCode)) {
|
|
alert(getNotification(oData.ErrorCode));
|
|
}
|
|
}
|
|
|
|
this.reloadMessageList(!MessageStore.messageList.length);
|
|
this.quotaDebounce();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} sFromFolderFullNameRaw
|
|
* @param {Array} aUidForRemove
|
|
*/
|
|
deleteMessagesFromFolderWithoutCheck(sFromFolderFullNameRaw, aUidForRemove) {
|
|
this.messagesDeleteHelper(sFromFolderFullNameRaw, aUidForRemove);
|
|
MessageStore.removeMessagesFromList(sFromFolderFullNameRaw, aUidForRemove);
|
|
}
|
|
|
|
/**
|
|
* @param {number} iDeleteType
|
|
* @param {string} sFromFolderFullNameRaw
|
|
* @param {Array} aUidForRemove
|
|
* @param {boolean=} bUseFolder = true
|
|
*/
|
|
deleteMessagesFromFolder(iDeleteType, sFromFolderFullNameRaw, aUidForRemove, bUseFolder) {
|
|
let oMoveFolder = null,
|
|
nSetSystemFoldersNotification = null;
|
|
|
|
switch (iDeleteType) {
|
|
case FolderType.Spam:
|
|
oMoveFolder = getFolderFromCacheList(FolderStore.spamFolder());
|
|
nSetSystemFoldersNotification = SetSystemFoldersNotification.Spam;
|
|
break;
|
|
case FolderType.NotSpam:
|
|
oMoveFolder = getFolderFromCacheList(getFolderInboxName());
|
|
break;
|
|
case FolderType.Trash:
|
|
oMoveFolder = getFolderFromCacheList(FolderStore.trashFolder());
|
|
nSetSystemFoldersNotification = SetSystemFoldersNotification.Trash;
|
|
break;
|
|
case FolderType.Archive:
|
|
oMoveFolder = getFolderFromCacheList(FolderStore.archiveFolder());
|
|
nSetSystemFoldersNotification = SetSystemFoldersNotification.Archive;
|
|
break;
|
|
// no default
|
|
}
|
|
|
|
bUseFolder = undefined === bUseFolder ? true : !!bUseFolder;
|
|
if (bUseFolder) {
|
|
if (
|
|
(FolderType.Spam === iDeleteType && UNUSED_OPTION_VALUE === FolderStore.spamFolder()) ||
|
|
(FolderType.Trash === iDeleteType && UNUSED_OPTION_VALUE === FolderStore.trashFolder()) ||
|
|
(FolderType.Archive === iDeleteType && UNUSED_OPTION_VALUE === FolderStore.archiveFolder())
|
|
) {
|
|
bUseFolder = false;
|
|
}
|
|
}
|
|
|
|
if (!oMoveFolder && bUseFolder) {
|
|
showScreenPopup(FolderSystemPopupView, [nSetSystemFoldersNotification]);
|
|
} else if (
|
|
!bUseFolder ||
|
|
(FolderType.Trash === iDeleteType &&
|
|
(sFromFolderFullNameRaw === FolderStore.spamFolder() || sFromFolderFullNameRaw === FolderStore.trashFolder()))
|
|
) {
|
|
showScreenPopup(AskPopupView, [
|
|
i18n('POPUPS_ASK/DESC_WANT_DELETE_MESSAGES'),
|
|
() => {
|
|
this.messagesDeleteHelper(sFromFolderFullNameRaw, aUidForRemove);
|
|
MessageStore.removeMessagesFromList(sFromFolderFullNameRaw, aUidForRemove);
|
|
}
|
|
]);
|
|
} else if (oMoveFolder) {
|
|
this.messagesMoveHelper(sFromFolderFullNameRaw, oMoveFolder.fullNameRaw, aUidForRemove);
|
|
MessageStore.removeMessagesFromList(sFromFolderFullNameRaw, aUidForRemove, oMoveFolder.fullNameRaw);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} sFromFolderFullNameRaw
|
|
* @param {Array} aUidForMove
|
|
* @param {string} sToFolderFullNameRaw
|
|
* @param {boolean=} bCopy = false
|
|
*/
|
|
moveMessagesToFolder(sFromFolderFullNameRaw, aUidForMove, sToFolderFullNameRaw, bCopy) {
|
|
if (sFromFolderFullNameRaw !== sToFolderFullNameRaw && Array.isArray(aUidForMove) && aUidForMove.length) {
|
|
const oFromFolder = getFolderFromCacheList(sFromFolderFullNameRaw),
|
|
oToFolder = getFolderFromCacheList(sToFolderFullNameRaw);
|
|
|
|
if (oFromFolder && oToFolder) {
|
|
if (undefined === bCopy ? false : !!bCopy) {
|
|
this.messagesCopyHelper(oFromFolder.fullNameRaw, oToFolder.fullNameRaw, aUidForMove);
|
|
} else {
|
|
this.messagesMoveHelper(oFromFolder.fullNameRaw, oToFolder.fullNameRaw, aUidForMove);
|
|
}
|
|
|
|
MessageStore.removeMessagesFromList(oFromFolder.fullNameRaw, aUidForMove, oToFolder.fullNameRaw, bCopy);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @param {Function=} callback = null
|
|
*/
|
|
foldersReload(callback = null) {
|
|
const prom = Remote.foldersReload(FolderStore.foldersLoading);
|
|
if (callback) {
|
|
prom
|
|
.then((value) => !!value)
|
|
.then(callback)
|
|
.catch(() => {
|
|
setTimeout(() => {
|
|
if (callback) {
|
|
callback(false); // eslint-disable-line callback-return
|
|
}
|
|
}, 1);
|
|
});
|
|
}
|
|
}
|
|
|
|
foldersPromisesActionHelper(promise, errorDefCode) {
|
|
Remote.abort('Folders')
|
|
.fastResolve(true)
|
|
.then(() => promise)
|
|
.then(
|
|
() => {
|
|
Remote.foldersReloadWithTimeout(FolderStore.foldersLoading);
|
|
},
|
|
(errorCode) => {
|
|
FolderStore.folderListError(getNotification(errorCode, '', errorDefCode));
|
|
Remote.foldersReloadWithTimeout(FolderStore.foldersLoading);
|
|
}
|
|
);
|
|
}
|
|
|
|
reloadOpenPgpKeys() {
|
|
if (PgpStore.capaOpenPGP()) {
|
|
const keys = [],
|
|
email = new EmailModel(),
|
|
openpgpKeyring = PgpStore.openpgpKeyring,
|
|
openpgpKeys = openpgpKeyring ? openpgpKeyring.getAllKeys() : [];
|
|
|
|
openpgpKeys.forEach((oItem, iIndex) => {
|
|
if (oItem && oItem.primaryKey) {
|
|
const aEmails = [],
|
|
aUsers = [],
|
|
primaryUser = oItem.getPrimaryUser(),
|
|
user =
|
|
primaryUser && primaryUser.user
|
|
? primaryUser.user.userId.userid
|
|
: oItem.users && oItem.users[0]
|
|
? oItem.users[0].userId.userid
|
|
: '';
|
|
|
|
if (oItem.users) {
|
|
oItem.users.forEach(item => {
|
|
if (item.userId) {
|
|
email.clear();
|
|
email.parse(item.userId.userid);
|
|
if (email.validate()) {
|
|
aEmails.push(email.email);
|
|
aUsers.push(item.userId.userid);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
if (aEmails.length) {
|
|
keys.push(
|
|
new OpenPgpKeyModel(
|
|
iIndex,
|
|
oItem.primaryKey.getFingerprint(),
|
|
oItem.primaryKey
|
|
.getKeyId()
|
|
.toHex()
|
|
.toLowerCase(),
|
|
oItem.getKeyIds()
|
|
.map(item => (item && item.toHex ? item.toHex() : null))
|
|
.validUnique(),
|
|
aUsers,
|
|
aEmails,
|
|
oItem.isPrivate(),
|
|
oItem.armor(),
|
|
user
|
|
)
|
|
);
|
|
}
|
|
}
|
|
});
|
|
|
|
delegateRunOnDestroy(PgpStore.openpgpkeys());
|
|
PgpStore.openpgpkeys(keys);
|
|
}
|
|
}
|
|
|
|
accountsAndIdentities() {
|
|
AccountStore.accounts.loading(true);
|
|
IdentityStore.identities.loading(true);
|
|
|
|
Remote.accountsAndIdentities((sResult, oData) => {
|
|
AccountStore.accounts.loading(false);
|
|
IdentityStore.identities.loading(false);
|
|
|
|
if (StorageResultType.Success === sResult && oData.Result) {
|
|
const counts = {},
|
|
sAccountEmail = AccountStore.email();
|
|
let parentEmail = Settings.get('ParentEmail') || sAccountEmail;
|
|
|
|
if (Array.isArray(oData.Result.Accounts)) {
|
|
AccountStore.accounts.forEach(oAccount =>
|
|
counts[oAccount.email] = oAccount.count()
|
|
);
|
|
|
|
delegateRunOnDestroy(AccountStore.accounts());
|
|
|
|
AccountStore.accounts(
|
|
oData.Result.Accounts.map(
|
|
sValue => new AccountModel(sValue, sValue !== parentEmail, counts[sValue] || 0)
|
|
)
|
|
);
|
|
}
|
|
|
|
if (Array.isArray(oData.Result.Identities)) {
|
|
delegateRunOnDestroy(IdentityStore.identities());
|
|
|
|
IdentityStore.identities(
|
|
oData.Result.Identities.map(identityData => {
|
|
const id = pString(identityData.Id),
|
|
email = pString(identityData.Email),
|
|
identity = new IdentityModel(id, email);
|
|
|
|
identity.name(pString(identityData.Name));
|
|
identity.replyTo(pString(identityData.ReplyTo));
|
|
identity.bcc(pString(identityData.Bcc));
|
|
identity.signature(pString(identityData.Signature));
|
|
identity.signatureInsertBefore(!!identityData.SignatureInsertBefore);
|
|
|
|
return identity;
|
|
})
|
|
);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
templates() {
|
|
TemplateStore.templates.loading(true);
|
|
|
|
Remote.templates((result, data) => {
|
|
TemplateStore.templates.loading(false);
|
|
|
|
if (StorageResultType.Success === result && data.Result && Array.isArray(data.Result.Templates)) {
|
|
delegateRunOnDestroy(TemplateStore.templates());
|
|
|
|
TemplateStore.templates(
|
|
data.Result.Templates.map(templateData =>
|
|
TemplateModel.reviveFromJson(templateData)
|
|
).filter(v => v)
|
|
);
|
|
}
|
|
});
|
|
}
|
|
|
|
quota() {
|
|
Remote.quota((result, data) => {
|
|
if (
|
|
StorageResultType.Success === result &&
|
|
data &&
|
|
data.Result &&
|
|
Array.isArray(data.Result) &&
|
|
1 < data.Result.length &&
|
|
isPosNumeric(data.Result[0], true) &&
|
|
isPosNumeric(data.Result[1], true)
|
|
) {
|
|
QuotaStore.populateData(pInt(data.Result[1]), pInt(data.Result[0]));
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @param {string} folder
|
|
* @param {Array=} list = []
|
|
*/
|
|
folderInformation(folder, list) {
|
|
if (folder && folder.trim()) {
|
|
Remote.folderInformation(
|
|
(result, data) => {
|
|
if (StorageResultType.Success === result) {
|
|
if (data && data.Result && data.Result.Hash && data.Result.Folder) {
|
|
let uid = '',
|
|
check = false,
|
|
unreadCountChange = false;
|
|
|
|
const folderFromCache = getFolderFromCacheList(data.Result.Folder);
|
|
if (folderFromCache) {
|
|
folderFromCache.interval = Date.now();
|
|
|
|
if (data.Result.Hash) {
|
|
setFolderHash(data.Result.Folder, data.Result.Hash);
|
|
}
|
|
|
|
if (null != data.Result.MessageCount) {
|
|
folderFromCache.messageCountAll(data.Result.MessageCount);
|
|
}
|
|
|
|
if (null != data.Result.MessageUnseenCount) {
|
|
if (pInt(folderFromCache.messageCountUnread()) !== pInt(data.Result.MessageUnseenCount)) {
|
|
unreadCountChange = true;
|
|
}
|
|
|
|
folderFromCache.messageCountUnread(data.Result.MessageUnseenCount);
|
|
}
|
|
|
|
if (unreadCountChange) {
|
|
MessageFlagsCache.clearFolder(folderFromCache.fullNameRaw);
|
|
}
|
|
|
|
if (data.Result.Flags) {
|
|
for (uid in data.Result.Flags) {
|
|
if (Object.prototype.hasOwnProperty.call(data.Result.Flags, uid)) {
|
|
check = true;
|
|
const flags = data.Result.Flags[uid];
|
|
MessageFlagsCache.storeByFolderAndUid(folderFromCache.fullNameRaw, uid.toString(), [
|
|
!!flags.IsUnseen,
|
|
!!flags.IsFlagged,
|
|
!!flags.IsAnswered,
|
|
!!flags.IsForwarded,
|
|
!!flags.IsReadReceipt
|
|
]);
|
|
}
|
|
}
|
|
|
|
if (check) {
|
|
this.reloadFlagsCurrentMessageListAndMessageFromCache();
|
|
}
|
|
}
|
|
|
|
MessageStore.initUidNextAndNewMessages(
|
|
folderFromCache.fullNameRaw,
|
|
data.Result.UidNext,
|
|
data.Result.NewMessages
|
|
);
|
|
|
|
const hash = getFolderHash(data.Result.Folder);
|
|
if (!hash || unreadCountChange || data.Result.Hash !== hash) {
|
|
if (folderFromCache.fullNameRaw === FolderStore.currentFolderFullNameRaw()) {
|
|
this.reloadMessageList();
|
|
} else if (getFolderInboxName() === folderFromCache.fullNameRaw) {
|
|
this.recacheInboxMessageList();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
folder,
|
|
list
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {boolean=} boot = false
|
|
*/
|
|
folderInformationMultiply(boot = false) {
|
|
const folders = FolderStore.getNextFolderNames();
|
|
if (Array.isNotEmpty(folders)) {
|
|
Remote.folderInformationMultiply((sResult, oData) => {
|
|
if (StorageResultType.Success === sResult) {
|
|
if (oData && oData.Result && oData.Result.List && Array.isNotEmpty(oData.Result.List)) {
|
|
const utc = Date.now();
|
|
oData.Result.List.forEach(item => {
|
|
const hash = getFolderHash(item.Folder),
|
|
folder = getFolderFromCacheList(item.Folder);
|
|
let unreadCountChange = false;
|
|
|
|
if (folder) {
|
|
folder.interval = utc;
|
|
|
|
if (item.Hash) {
|
|
setFolderHash(item.Folder, item.Hash);
|
|
}
|
|
|
|
if (null != item.MessageCount) {
|
|
folder.messageCountAll(item.MessageCount);
|
|
}
|
|
|
|
if (null != item.MessageUnseenCount) {
|
|
if (pInt(folder.messageCountUnread()) !== pInt(item.MessageUnseenCount)) {
|
|
unreadCountChange = true;
|
|
}
|
|
|
|
folder.messageCountUnread(item.MessageUnseenCount);
|
|
}
|
|
|
|
if (unreadCountChange) {
|
|
MessageFlagsCache.clearFolder(folder.fullNameRaw);
|
|
}
|
|
|
|
if (!hash || item.Hash !== hash) {
|
|
if (folder.fullNameRaw === FolderStore.currentFolderFullNameRaw()) {
|
|
this.reloadMessageList();
|
|
}
|
|
} else if (unreadCountChange
|
|
&& folder.fullNameRaw === FolderStore.currentFolderFullNameRaw()
|
|
&& MessageStore.messageList.length) {
|
|
this.folderInformation(folder.fullNameRaw, MessageStore.messageList());
|
|
}
|
|
}
|
|
});
|
|
|
|
if (boot) {
|
|
setTimeout(() => this.folderInformationMultiply(true), 2000);
|
|
}
|
|
}
|
|
}
|
|
}, folders);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} sFolderFullNameRaw
|
|
* @param {number} iSetAction
|
|
* @param {Array=} messages = null
|
|
*/
|
|
messageListAction(sFolderFullNameRaw, iSetAction, messages) {
|
|
let folder = null,
|
|
alreadyUnread = 0,
|
|
rootUids = [];
|
|
|
|
if (undefined === messages || !messages) {
|
|
messages = MessageStore.messageListChecked();
|
|
}
|
|
|
|
rootUids = messages.map(oMessage => oMessage && oMessage.uid ? oMessage.uid : null)
|
|
.validUnique();
|
|
|
|
if (sFolderFullNameRaw && rootUids.length) {
|
|
switch (iSetAction) {
|
|
case MessageSetAction.SetSeen:
|
|
rootUids.forEach(sSubUid =>
|
|
alreadyUnread += MessageFlagsCache.storeBySetAction(sFolderFullNameRaw, sSubUid, iSetAction)
|
|
);
|
|
|
|
folder = getFolderFromCacheList(sFolderFullNameRaw);
|
|
if (folder) {
|
|
folder.messageCountUnread(folder.messageCountUnread() - alreadyUnread);
|
|
}
|
|
|
|
Remote.messageSetSeen(()=>{}, sFolderFullNameRaw, rootUids, true);
|
|
break;
|
|
|
|
case MessageSetAction.UnsetSeen:
|
|
rootUids.forEach(sSubUid =>
|
|
alreadyUnread += MessageFlagsCache.storeBySetAction(sFolderFullNameRaw, sSubUid, iSetAction)
|
|
);
|
|
|
|
folder = getFolderFromCacheList(sFolderFullNameRaw);
|
|
if (folder) {
|
|
folder.messageCountUnread(folder.messageCountUnread() - alreadyUnread + rootUids.length);
|
|
}
|
|
|
|
Remote.messageSetSeen(()=>{}, sFolderFullNameRaw, rootUids, false);
|
|
break;
|
|
|
|
case MessageSetAction.SetFlag:
|
|
rootUids.forEach(sSubUid =>
|
|
MessageFlagsCache.storeBySetAction(sFolderFullNameRaw, sSubUid, iSetAction)
|
|
);
|
|
|
|
Remote.messageSetFlagged(()=>{}, sFolderFullNameRaw, rootUids, true);
|
|
break;
|
|
|
|
case MessageSetAction.UnsetFlag:
|
|
rootUids.forEach(sSubUid =>
|
|
MessageFlagsCache.storeBySetAction(sFolderFullNameRaw, sSubUid, iSetAction)
|
|
);
|
|
|
|
Remote.messageSetFlagged(()=>{}, sFolderFullNameRaw, rootUids, false);
|
|
break;
|
|
// no default
|
|
}
|
|
|
|
this.reloadFlagsCurrentMessageListAndMessageFromCache();
|
|
MessageStore.messageViewTrigger(!MessageStore.messageViewTrigger());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} query
|
|
* @param {Function} autocompleteCallback
|
|
*/
|
|
getAutocomplete(query, autocompleteCallback) {
|
|
Remote.suggestions((result, data) => {
|
|
if (StorageResultType.Success === result && data && Array.isArray(data.Result)) {
|
|
autocompleteCallback(
|
|
data.Result.map(item => (item && item[0] ? new EmailModel(item[0], item[1]) : null)).filter(v => v)
|
|
);
|
|
} else if (StorageResultType.Abort !== result) {
|
|
autocompleteCallback([]);
|
|
}
|
|
}, query);
|
|
}
|
|
|
|
/**
|
|
* @param {string} sFullNameHash
|
|
* @param {boolean} bExpanded
|
|
*/
|
|
setExpandedFolder(sFullNameHash, bExpanded) {
|
|
let aExpandedList = Local.get(ClientSideKeyName.ExpandedFolders);
|
|
if (!Array.isArray(aExpandedList)) {
|
|
aExpandedList = [];
|
|
}
|
|
|
|
if (bExpanded) {
|
|
if (!aExpandedList.includes(sFullNameHash))
|
|
aExpandedList.push(sFullNameHash);
|
|
} else {
|
|
aExpandedList = aExpandedList.filter(value => value !== sFullNameHash);
|
|
}
|
|
|
|
Local.set(ClientSideKeyName.ExpandedFolders, aExpandedList);
|
|
}
|
|
|
|
setLayoutResizer(source, target, sClientSideKeyName, mode) {
|
|
if (mode) {
|
|
source.classList.add('resizable');
|
|
if (!source.querySelector('.resizer')) {
|
|
const resizer = doc.createElement('div'),
|
|
cssint = s => parseFloat(getComputedStyle(source, null).getPropertyValue(s).replace('px', ''));
|
|
resizer.className = 'resizer';
|
|
source.append(resizer);
|
|
resizer.addEventListener('mousedown', {
|
|
source: source,
|
|
mode: mode,
|
|
handleEvent: function(e) {
|
|
if ('mousedown' == e.type) {
|
|
e.preventDefault();
|
|
this.pos = ('width' == this.mode) ? e.pageX : e.pageY;
|
|
this.min = cssint('min-'+this.mode);
|
|
this.max = cssint('max-'+this.mode);
|
|
this.org = cssint(this.mode);
|
|
addEventListener('mousemove', this);
|
|
addEventListener('mouseup', this);
|
|
} else if ('mousemove' == e.type) {
|
|
const length = this.org + (('width' == this.mode ? e.pageX : e.pageY) - this.pos);
|
|
if (length >= this.min && length <= this.max ) {
|
|
this.source.style[this.mode] = length + 'px';
|
|
}
|
|
} else if ('mouseup' == e.type) {
|
|
removeEventListener('mousemove', this);
|
|
removeEventListener('mouseup', this);
|
|
}
|
|
}
|
|
});
|
|
if ('width' == mode) {
|
|
source.observer = new ResizeObserver(() => {
|
|
target.style.left = source.offsetWidth + 'px';
|
|
Local.set(sClientSideKeyName, source.offsetWidth);
|
|
});
|
|
} else {
|
|
source.observer = new ResizeObserver(() => {
|
|
target.style.top = (4 + source.offsetTop + source.offsetHeight) + 'px';
|
|
Local.set(sClientSideKeyName, source.offsetHeight);
|
|
});
|
|
}
|
|
}
|
|
source.observer.observe(source, { box: 'border-box' });
|
|
const length = Local.get(sClientSideKeyName);
|
|
if (length) {
|
|
if ('width' == mode) {
|
|
source.style.width = length + 'px';
|
|
} else {
|
|
source.style.height = length + 'px';
|
|
}
|
|
}
|
|
} else {
|
|
source.observer && source.observer.disconnect();
|
|
source.classList.remove('resizable');
|
|
target.removeAttribute('style');
|
|
source.removeAttribute('style');
|
|
}
|
|
}
|
|
|
|
initHorizontalLayoutResizer(sClientSideKeyName) {
|
|
const top = doc.querySelector('.b-message-list-wrapper'),
|
|
bottom = doc.querySelector('.b-message-view-wrapper'),
|
|
fDisable = bDisable => {
|
|
this.setLayoutResizer(top, bottom, sClientSideKeyName,
|
|
(bDisable || !$htmlCL.contains('rl-bottom-preview-pane')) ? null : 'height');
|
|
};
|
|
if (top && bottom) {
|
|
fDisable(false);
|
|
addEventListener('rl-layout', e => fDisable(Layout.BottomPreview !== e.detail));
|
|
}
|
|
}
|
|
|
|
initVerticalLayoutResizer(sClientSideKeyName) {
|
|
const left = doc.getElementById('rl-left'),
|
|
right = doc.getElementById('rl-right'),
|
|
fDisable = bDisable => {
|
|
this.setLayoutResizer(left, right, sClientSideKeyName, bDisable ? null : 'width');
|
|
};
|
|
if (left && right) {
|
|
fDisable(false);
|
|
leftPanelDisabled.subscribe(value => fDisable(value));
|
|
}
|
|
}
|
|
|
|
logout() {
|
|
Remote.logout(() => this.logoutReload(0 < (Settings.get('ParentEmail')||{length:0}).length));
|
|
}
|
|
|
|
bootend() {
|
|
progressJs.end();
|
|
this.hideLoading();
|
|
}
|
|
|
|
bootstart() {
|
|
super.bootstart();
|
|
|
|
AppStore.populate();
|
|
SettingsStore.populate();
|
|
NotificationStore.populate();
|
|
AccountStore.populate();
|
|
ContactStore.populate();
|
|
|
|
let contactsSyncInterval = pInt(Settings.get('ContactsSyncInterval'));
|
|
|
|
const startupUrl = pString(Settings.get('StartupUrl'));
|
|
|
|
progressJs.set(90);
|
|
|
|
rl.setWindowTitle();
|
|
if (Settings.get('Auth')) {
|
|
if (
|
|
Settings.capa(Capa.TwoFactor) &&
|
|
Settings.capa(Capa.TwoFactorForce) &&
|
|
Settings.get('RequireTwoFactor')
|
|
) {
|
|
this.bootend();
|
|
showScreenPopup(TwoFactorConfigurationPopupView, [true]);
|
|
} else {
|
|
rl.setWindowTitle(i18n('GLOBAL/LOADING'));
|
|
|
|
// require.ensure([], function() { // require code splitting
|
|
|
|
this.foldersReload(value => {
|
|
try {
|
|
this.bootend();
|
|
|
|
if (value) {
|
|
if (startupUrl) {
|
|
rl.route.setHash(root(startupUrl), true);
|
|
}
|
|
|
|
if (window.crypto && crypto.getRandomValues && Settings.capa(Capa.OpenPGP)) {
|
|
const openpgpCallback = () => {
|
|
if (!window.openpgp) {
|
|
return false;
|
|
}
|
|
PgpStore.openpgp = openpgp;
|
|
|
|
if (window.Worker) {
|
|
try {
|
|
PgpStore.openpgp.initWorker({ path: openPgpWorkerJs() });
|
|
} catch (e) {
|
|
console.error(e);
|
|
}
|
|
}
|
|
|
|
PgpStore.openpgpKeyring = new openpgp.Keyring();
|
|
PgpStore.capaOpenPGP(true);
|
|
|
|
this.reloadOpenPgpKeys();
|
|
|
|
return true;
|
|
};
|
|
|
|
if (!openpgpCallback()) {
|
|
const script = doc.createElement('script');
|
|
script.onload = openpgpCallback;
|
|
script.onerror = () => console.error(script.src);
|
|
script.src = openPgpJs();
|
|
doc.head.append(script);
|
|
}
|
|
} else {
|
|
PgpStore.capaOpenPGP(false);
|
|
}
|
|
|
|
startScreens([
|
|
MailBoxUserScreen,
|
|
Settings.capa(Capa.Settings) ? SettingsUserScreen : null
|
|
// false ? AboutUserScreen : null
|
|
]);
|
|
|
|
setInterval(() => this.folderInformation(getFolderInboxName()), 120000);
|
|
setInterval(() => {
|
|
const sF = FolderStore.currentFolderFullNameRaw();
|
|
if (getFolderInboxName() !== sF) {
|
|
this.folderInformation(sF);
|
|
}
|
|
}, 180000);
|
|
|
|
setTimeout(() => setInterval(this.folderInformationMultiply, 120000), 300000);
|
|
setInterval(this.quota, 900000);
|
|
setInterval(this.foldersReload, 1200000);
|
|
|
|
setTimeout(this.contactsSync, 10000);
|
|
contactsSyncInterval = 5 <= contactsSyncInterval ? contactsSyncInterval : 20;
|
|
contactsSyncInterval = 320 >= contactsSyncInterval ? contactsSyncInterval : 320;
|
|
setInterval(this.contactsSync, contactsSyncInterval * 60000 + 5000);
|
|
|
|
setTimeout(() => this.folderInformationMultiply(true), 2000);
|
|
|
|
|
|
this.accountsAndIdentities(true);
|
|
|
|
setTimeout(() => {
|
|
const sF = FolderStore.currentFolderFullNameRaw();
|
|
if (getFolderInboxName() !== sF) {
|
|
this.folderInformation(sF);
|
|
}
|
|
}, 1000);
|
|
|
|
setTimeout(this.quota, 5000);
|
|
setTimeout(() => Remote.appDelayStart(()=>{}), 35000);
|
|
|
|
addEventListener('rl.auto-logout', () => this.logout());
|
|
|
|
if (
|
|
!!Settings.get('AccountSignMe') &&
|
|
navigator.registerProtocolHandler &&
|
|
Settings.capa(Capa.Composer)
|
|
) {
|
|
setTimeout(() => {
|
|
try {
|
|
navigator.registerProtocolHandler(
|
|
'mailto',
|
|
location.protocol + '//' + location.host + location.pathname + '?mailto&to=%s',
|
|
'' + (Settings.get('Title') || 'SnappyMail')
|
|
);
|
|
} catch (e) {} // eslint-disable-line no-empty
|
|
|
|
if (Settings.get('MailToEmail')) {
|
|
mailToHelper(Settings.get('MailToEmail'), ComposePopupView);
|
|
}
|
|
}, 500);
|
|
}
|
|
|
|
if (!Settings.app('mobile')) {
|
|
setTimeout(() => this.initVerticalLayoutResizer(ClientSideKeyName.FolderListSize), 1);
|
|
}
|
|
} else {
|
|
this.logout();
|
|
}
|
|
} catch (e) {
|
|
console.error(e);
|
|
}
|
|
});
|
|
|
|
// }); // require code splitting
|
|
}
|
|
} else {
|
|
this.bootend();
|
|
startScreens([LoginUserScreen]);
|
|
}
|
|
|
|
setInterval(() => dispatchEvent(new CustomEvent('reload-time')), 60000);
|
|
}
|
|
|
|
showComposePopupView(params = [])
|
|
{
|
|
showScreenPopup(ComposePopupView, params);
|
|
}
|
|
}
|
|
|
|
export default new AppUser();
|