import { isArray, arrayLength, pString, pInt } from 'Common/Utils'; import { getFolderHash, getFolderInboxName, getFolderUidNext, getFolderFromCacheList, MessageFlagsCache } from 'Common/Cache'; import { SettingsGet } from 'Common/Globals'; import { SUB_QUERY_PREFIX } from 'Common/Links'; import { AppUserStore } from 'Stores/User/App'; import { SettingsUserStore } from 'Stores/User/Settings'; import { FolderUserStore } from 'Stores/User/Folder'; import { AbstractFetchRemote } from 'Remote/AbstractFetch'; import { FolderCollectionModel } from 'Model/FolderCollection'; // unescape(encodeURIComponent()) makes the UTF-16 DOMString to an UTF-8 string const urlSafeJSON = data => btoa(unescape(encodeURIComponent(JSON.stringify(data)))) .replace(/\+/g, '-') .replace(/\//g, '_') .replace(/=+$/, ''); /* Withous deprecated 'unescape': const urlSafeJSON = data => btoa(encodeURIComponent(JSON.stringify(data)).replace( /%([0-9A-F]{2})/g, (match, p1) => String.fromCharCode('0x' + p1) )) .replace(/\+/g, '-') .replace(/\//g, '_') .replace(/=+$/, ''); */ class RemoteUserFetch extends AbstractFetchRemote { /** * @param {Function} fCallback * @param {object} params * @param {boolean=} bSilent = false */ messageList(fCallback, params, bSilent = false) { const sFolderFullName = pString(params.Folder), folderHash = getFolderHash(sFolderFullName); params = Object.assign({ Offset: 0, Limit: SettingsUserStore.messagesPerPage(), Search: '', UidNext: getFolderInboxName() === sFolderFullName ? getFolderUidNext(sFolderFullName) : '', Sort: FolderUserStore.sortMode(), Hash: folderHash + SettingsGet('AccountHash') }, params); params.Folder = sFolderFullName; if (AppUserStore.threadsAllowed() && SettingsUserStore.useThreads()) { params.UseThreads = 1; } else { params.ThreadUid = 0; } let sGetAdd = ''; if (folderHash && (!params.Search || !params.Search.includes('is:'))) { sGetAdd = 'MessageList/' + SUB_QUERY_PREFIX + '/' + urlSafeJSON(params); params = {}; } this.request('MessageList', fCallback, params, 30000, sGetAdd, bSilent ? [] : ['MessageList'] ); } /** * @param {?Function} fCallback * @param {string} sFolderFullName * @param {number} iUid * @returns {boolean} */ message(fCallback, sFolderFullName, iUid) { sFolderFullName = pString(sFolderFullName); iUid = pInt(iUid); if (getFolderFromCacheList(sFolderFullName) && 0 < iUid) { this.request('Message', fCallback, {}, null, 'Message/' + SUB_QUERY_PREFIX + '/' + urlSafeJSON([ sFolderFullName, iUid, AppUserStore.threadsAllowed() && SettingsUserStore.useThreads() ? 1 : 0, SettingsGet('AccountHash') ]), ['Message'] ); return true; } return false; } /** * @param {?Function} fCallback * @param {string} folder * @param {Array=} list = [] */ folderInformation(fCallback, folder, list = []) { let request = true; const uids = []; if (arrayLength(list)) { request = false; list.forEach(messageListItem => { if (!MessageFlagsCache.getFor(messageListItem.folder, messageListItem.uid)) { uids.push(messageListItem.uid); } if (messageListItem.threads.length) { messageListItem.threads.forEach(uid => { if (!MessageFlagsCache.getFor(messageListItem.folder, uid)) { uids.push(uid); } }); } }); if (uids.length) { request = true; } } if (request) { this.request('FolderInformation', fCallback, { Folder: folder, FlagsUids: isArray(uids) ? uids : [], UidNext: getFolderInboxName() === folder ? getFolderUidNext(folder) : 0 }); } else if (SettingsUserStore.useThreads()) { rl.app.reloadFlagsCurrentMessageListAndMessageFromCache(); } } /** * @param {?Function} fCallback * @param {string} sFolderFullName * @param {boolean} bSetSeen * @param {Array} aThreadUids = null */ messageSetSeenToAll(fCallback, sFolderFullName, bSetSeen, aThreadUids = null) { this.request('MessageSetSeenToAll', fCallback, { Folder: sFolderFullName, SetAction: bSetSeen ? 1 : 0, ThreadUids: aThreadUids ? aThreadUids.join(',') : '' }); } /** * @param {?Function} fCallback * @param {Object} oData */ saveSettings(fCallback, oData) { this.request('SettingsUpdate', fCallback, oData); } /** * @param {string} key * @param {?scalar} value * @param {?Function} fCallback */ saveSetting(key, value, fCallback) { this.saveSettings(fCallback, { [key]: value }); } /** * @param {?Function} fCallback * @param {string} sFolderFullName * @param {boolean} bSubscribe */ folderSetMetadata(fCallback, sFolderFullName, sKey, sValue) { this.request('FolderSetMetadata', fCallback, { Folder: sFolderFullName, Key: sKey, Value: sValue }); } /** * @param {?Function} fCallback * @param {string} sQuery * @param {number} iPage */ suggestions(fCallback, sQuery, iPage) { this.request('Suggestions', fCallback, { Query: sQuery, Page: iPage }, null, '', ['Suggestions'] ); } /** * @param {?Function} fCallback */ foldersReload(fCallback) { this.abort('Folders') .post('Folders', FolderUserStore.foldersLoading) .then(data => { data = FolderCollectionModel.reviveFromJson(data.Result); data && data.storeIt(); fCallback && fCallback(true); }) .catch(() => fCallback && setTimeout(() => fCallback(false), 1)); } foldersReloadWithTimeout() { this.setTrigger(FolderUserStore.foldersLoading, true); clearTimeout(this.foldersTimeout); this.foldersTimeout = setTimeout(() => this.foldersReload(), 500); } /* folderMove(sPrevFolderFullName, sNewFolderFullName) { return this.post('FolderMove', FolderUserStore.foldersRenaming, { Folder: sPrevFolderFullName, NewFolder: sNewFolderFullName }); } */ attachmentsActions(sAction, aHashes, fTrigger) { return this.post('AttachmentsActions', fTrigger, { Do: sAction, Hashes: aHashes }); } } export default new RemoteUserFetch();