2022-06-07 18:00:22 +08:00
|
|
|
import { MessageFlagsCache } from 'Common/Cache';
|
2022-02-24 19:43:44 +08:00
|
|
|
import { Notification } from 'Common/Enums';
|
2022-02-24 02:26:52 +08:00
|
|
|
import { MessageSetAction, ComposeType/*, FolderType*/ } from 'Common/EnumsUser';
|
2022-11-22 18:14:56 +08:00
|
|
|
import { doc, createElement, elementById, dropdowns, dropdownVisibility, SettingsGet, leftPanelDisabled } from 'Common/Globals';
|
2022-02-24 02:26:52 +08:00
|
|
|
import { plainToHtml } from 'Common/Html';
|
|
|
|
import { getNotification } from 'Common/Translator';
|
2021-01-26 05:00:13 +08:00
|
|
|
import { EmailModel } from 'Model/Email';
|
2022-02-24 02:26:52 +08:00
|
|
|
import { MessageModel } from 'Model/Message';
|
|
|
|
import { MessageUserStore } from 'Stores/User/Message';
|
|
|
|
import { MessagelistUserStore } from 'Stores/User/Messagelist';
|
2021-08-19 21:14:47 +08:00
|
|
|
import { SettingsUserStore } from 'Stores/User/Settings';
|
2021-11-05 17:20:06 +08:00
|
|
|
import * as Local from 'Storage/Client';
|
2022-02-10 23:35:55 +08:00
|
|
|
import { ThemeStore } from 'Stores/Theme';
|
2022-02-24 06:11:12 +08:00
|
|
|
import Remote from 'Remote/User/Fetch';
|
2023-01-26 17:41:55 +08:00
|
|
|
import { attachmentDownload } from 'Common/Links';
|
2020-10-15 01:16:37 +08:00
|
|
|
|
2021-11-05 17:20:06 +08:00
|
|
|
export const
|
2021-12-09 18:24:30 +08:00
|
|
|
|
2022-11-22 18:14:56 +08:00
|
|
|
moveAction = ko.observable(false),
|
|
|
|
|
2022-03-08 17:52:08 +08:00
|
|
|
dropdownsDetectVisibility = (() =>
|
|
|
|
dropdownVisibility(!!dropdowns.find(item => item.classList.contains('show')))
|
|
|
|
).debounce(50),
|
|
|
|
|
2022-02-10 23:35:55 +08:00
|
|
|
/**
|
|
|
|
* @param {string} link
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
download = (link, name = "") => {
|
2022-05-19 16:58:04 +08:00
|
|
|
console.log('download: '+link);
|
2022-03-24 18:28:01 +08:00
|
|
|
// Firefox 98 issue https://github.com/the-djmaze/snappymail/issues/301
|
|
|
|
if (ThemeStore.isMobile() || /firefox/i.test(navigator.userAgent)) {
|
|
|
|
open(link, '_blank');
|
2022-02-10 23:35:55 +08:00
|
|
|
focus();
|
|
|
|
} else {
|
2022-03-24 18:28:01 +08:00
|
|
|
const oLink = createElement('a', {
|
|
|
|
href: link,
|
|
|
|
target: '_blank',
|
|
|
|
download: name
|
|
|
|
});
|
2022-02-10 23:35:55 +08:00
|
|
|
doc.body.appendChild(oLink).click();
|
|
|
|
oLink.remove();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2023-01-26 17:41:55 +08:00
|
|
|
downloadZip = (hashes, onError, fTrigger, folder) => {
|
|
|
|
if (hashes.length) {
|
|
|
|
let params = {
|
|
|
|
target: 'zip',
|
|
|
|
hashes: hashes
|
|
|
|
};
|
|
|
|
if (!onError) {
|
|
|
|
onError = () => alert('Download failed');
|
|
|
|
}
|
|
|
|
if (folder) {
|
|
|
|
params.folder = folder;
|
|
|
|
// params.uids = uids;
|
|
|
|
}
|
|
|
|
Remote.post('AttachmentsActions', fTrigger || null, params)
|
|
|
|
.then(result => {
|
|
|
|
let hash = result?.Result?.fileHash;
|
|
|
|
hash ? download(attachmentDownload(hash), hash+'.zip') : onError();
|
|
|
|
})
|
|
|
|
.catch(onError);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-10-15 01:16:37 +08:00
|
|
|
/**
|
|
|
|
* @returns {function}
|
|
|
|
*/
|
2021-11-05 17:20:06 +08:00
|
|
|
computedPaginatorHelper = (koCurrentPage, koPageCount) => {
|
2020-10-15 01:16:37 +08:00
|
|
|
return () => {
|
|
|
|
const currentPage = koCurrentPage(),
|
|
|
|
pageCount = koPageCount(),
|
|
|
|
result = [],
|
|
|
|
fAdd = (index, push = true, customName = '') => {
|
|
|
|
const data = {
|
|
|
|
current: index === currentPage,
|
|
|
|
name: customName ? customName.toString() : index.toString(),
|
|
|
|
custom: !!customName,
|
|
|
|
title: customName ? index.toString() : '',
|
|
|
|
value: index.toString()
|
|
|
|
};
|
|
|
|
|
2022-10-10 19:52:56 +08:00
|
|
|
push ? result.push(data) : result.unshift(data);
|
2020-10-15 01:16:37 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
let prev = 0,
|
|
|
|
next = 0,
|
|
|
|
limit = 2;
|
|
|
|
|
|
|
|
if (1 < pageCount || (0 < pageCount && pageCount < currentPage)) {
|
|
|
|
if (pageCount < currentPage) {
|
|
|
|
fAdd(pageCount);
|
|
|
|
prev = pageCount;
|
|
|
|
next = pageCount;
|
|
|
|
} else {
|
|
|
|
if (3 >= currentPage || pageCount - 2 <= currentPage) {
|
|
|
|
limit += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
fAdd(currentPage);
|
|
|
|
prev = currentPage;
|
|
|
|
next = currentPage;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (0 < limit) {
|
|
|
|
--prev;
|
|
|
|
++next;
|
|
|
|
|
|
|
|
if (0 < prev) {
|
|
|
|
fAdd(prev, false);
|
|
|
|
--limit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pageCount >= next) {
|
|
|
|
fAdd(next, true);
|
|
|
|
--limit;
|
|
|
|
} else if (0 >= prev) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (3 === prev) {
|
|
|
|
fAdd(2, false);
|
|
|
|
} else if (3 < prev) {
|
2021-07-22 22:24:24 +08:00
|
|
|
fAdd(Math.round((prev - 1) / 2), false, '…');
|
2020-10-15 01:16:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (pageCount - 2 === next) {
|
|
|
|
fAdd(pageCount - 1, true);
|
|
|
|
} else if (pageCount - 2 > next) {
|
2021-07-22 22:24:24 +08:00
|
|
|
fAdd(Math.round((pageCount + next) / 2), true, '…');
|
2020-10-15 01:16:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// first and last
|
|
|
|
if (1 < prev) {
|
|
|
|
fAdd(1, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pageCount > next) {
|
|
|
|
fAdd(pageCount, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
};
|
2021-11-05 17:20:06 +08:00
|
|
|
},
|
2020-10-15 01:16:37 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} mailToUrl
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
2022-03-08 17:05:24 +08:00
|
|
|
mailToHelper = mailToUrl => {
|
2022-09-02 17:52:07 +08:00
|
|
|
if ('mailto:' === mailToUrl?.slice(0, 7).toLowerCase()) {
|
2022-03-08 17:05:24 +08:00
|
|
|
mailToUrl = mailToUrl.slice(7).split('?');
|
|
|
|
|
|
|
|
const
|
|
|
|
email = mailToUrl[0],
|
|
|
|
params = new URLSearchParams(mailToUrl[1]),
|
|
|
|
toEmailModel = value => null != value ? EmailModel.parseEmailLine(value) : null;
|
2020-10-15 01:16:37 +08:00
|
|
|
|
2021-03-10 18:43:23 +08:00
|
|
|
showMessageComposer([
|
2020-10-15 01:16:37 +08:00
|
|
|
ComposeType.Empty,
|
|
|
|
null,
|
2022-03-08 17:05:24 +08:00
|
|
|
params.get('to')
|
|
|
|
? Object.values(
|
|
|
|
toEmailModel(email + ',' + params.get('to')).reduce((result, value) => {
|
|
|
|
if (value) {
|
|
|
|
if (result[value.email]) {
|
|
|
|
if (!result[value.email].name) {
|
|
|
|
result[value.email] = value;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
result[value.email] = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}, {})
|
|
|
|
)
|
|
|
|
: EmailModel.parseEmailLine(email),
|
|
|
|
toEmailModel(params.get('cc')),
|
|
|
|
toEmailModel(params.get('bcc')),
|
|
|
|
params.get('subject'),
|
|
|
|
plainToHtml(params.get('body') || '')
|
2020-10-15 01:16:37 +08:00
|
|
|
]);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2021-11-05 17:20:06 +08:00
|
|
|
},
|
2021-03-10 18:43:23 +08:00
|
|
|
|
2021-11-05 17:20:06 +08:00
|
|
|
showMessageComposer = (params = []) =>
|
2021-03-10 18:43:23 +08:00
|
|
|
{
|
2021-03-10 18:44:48 +08:00
|
|
|
rl.app.showMessageComposer(params);
|
2021-11-05 17:20:06 +08:00
|
|
|
},
|
|
|
|
|
|
|
|
setLayoutResizer = (source, target, sClientSideKeyName, mode) =>
|
|
|
|
{
|
|
|
|
if (source.layoutResizer && source.layoutResizer.mode != mode) {
|
2023-01-26 21:11:30 +08:00
|
|
|
target?.removeAttribute('style');
|
2021-11-05 17:20:06 +08:00
|
|
|
source.removeAttribute('style');
|
|
|
|
}
|
2022-09-02 17:52:07 +08:00
|
|
|
source.observer?.disconnect();
|
2021-11-06 19:27:37 +08:00
|
|
|
// source.classList.toggle('resizable', mode);
|
2021-11-05 17:20:06 +08:00
|
|
|
if (mode) {
|
2022-06-08 23:14:44 +08:00
|
|
|
const length = Local.get(sClientSideKeyName + mode) || SettingsGet('Resizer' + sClientSideKeyName + mode),
|
2022-06-09 05:40:28 +08:00
|
|
|
setTargetPos = mode => {
|
2022-06-08 23:14:44 +08:00
|
|
|
let value;
|
2022-06-09 05:40:28 +08:00
|
|
|
if ('Width' == mode) {
|
2022-06-08 23:14:44 +08:00
|
|
|
value = source.offsetWidth;
|
2023-01-26 21:11:30 +08:00
|
|
|
target && (target.style.left = value + 'px');
|
2022-06-08 23:14:44 +08:00
|
|
|
} else {
|
|
|
|
value = source.offsetHeight;
|
2023-01-26 21:11:30 +08:00
|
|
|
target && (target.style.top = value + 'px');
|
2022-06-08 23:14:44 +08:00
|
|
|
}
|
|
|
|
return value;
|
|
|
|
};
|
2022-06-09 05:40:28 +08:00
|
|
|
if (length) {
|
|
|
|
source.style[mode.toLowerCase()] = length + 'px';
|
|
|
|
setTargetPos(mode);
|
|
|
|
}
|
2021-11-05 17:20:06 +08:00
|
|
|
if (!source.layoutResizer) {
|
|
|
|
const resizer = createElement('div', {'class':'resizer'}),
|
2022-06-09 05:40:28 +08:00
|
|
|
save = (data => Remote.saveSettings(0, data)).debounce(500),
|
2021-11-05 17:20:06 +08:00
|
|
|
size = {},
|
2021-11-06 19:27:37 +08:00
|
|
|
store = () => {
|
2022-06-09 05:40:28 +08:00
|
|
|
const value = setTargetPos(resizer.mode),
|
2022-06-08 23:14:44 +08:00
|
|
|
prop = resizer.key + resizer.mode;
|
2022-06-09 05:40:28 +08:00
|
|
|
(value == Local.get(prop)) || Local.set(prop, value);
|
|
|
|
(value == SettingsGet('Resizer' + prop)) || save({['Resizer' + prop]: value});
|
2021-11-06 19:27:37 +08:00
|
|
|
},
|
2021-11-05 17:20:06 +08:00
|
|
|
cssint = s => {
|
|
|
|
let value = getComputedStyle(source, null)[s].replace('px', '');
|
|
|
|
if (value.includes('%')) {
|
2021-11-06 19:27:37 +08:00
|
|
|
value = source.parentElement['offset'+resizer.mode]
|
2021-11-05 17:20:06 +08:00
|
|
|
* value.replace('%', '') / 100;
|
|
|
|
}
|
|
|
|
return parseFloat(value);
|
|
|
|
};
|
|
|
|
source.layoutResizer = resizer;
|
|
|
|
source.append(resizer);
|
|
|
|
resizer.addEventListener('mousedown', {
|
|
|
|
handleEvent: function(e) {
|
|
|
|
if ('mousedown' == e.type) {
|
2021-11-06 19:27:37 +08:00
|
|
|
const lmode = resizer.mode.toLowerCase();
|
2021-11-05 17:20:06 +08:00
|
|
|
e.preventDefault();
|
2021-11-06 19:27:37 +08:00
|
|
|
size.pos = ('width' == lmode) ? e.pageX : e.pageY;
|
|
|
|
size.min = cssint('min-'+lmode);
|
|
|
|
size.max = cssint('max-'+lmode);
|
|
|
|
size.org = cssint(lmode);
|
2021-11-05 17:20:06 +08:00
|
|
|
addEventListener('mousemove', this);
|
|
|
|
addEventListener('mouseup', this);
|
|
|
|
} else if ('mousemove' == e.type) {
|
2021-11-06 19:27:37 +08:00
|
|
|
const lmode = resizer.mode.toLowerCase(),
|
|
|
|
length = size.org + (('width' == lmode ? e.pageX : e.pageY) - size.pos);
|
2021-11-05 17:20:06 +08:00
|
|
|
if (length >= size.min && length <= size.max ) {
|
2021-11-06 19:27:37 +08:00
|
|
|
source.style[lmode] = length + 'px';
|
|
|
|
source.observer || store();
|
2021-11-05 17:20:06 +08:00
|
|
|
}
|
|
|
|
} else if ('mouseup' == e.type) {
|
|
|
|
removeEventListener('mousemove', this);
|
|
|
|
removeEventListener('mouseup', this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2021-11-06 19:27:37 +08:00
|
|
|
source.observer = window.ResizeObserver ? new ResizeObserver(store) : null;
|
2021-11-05 17:20:06 +08:00
|
|
|
}
|
|
|
|
source.layoutResizer.mode = mode;
|
2021-11-06 19:27:37 +08:00
|
|
|
source.layoutResizer.key = sClientSideKeyName;
|
2022-09-02 17:52:07 +08:00
|
|
|
source.observer?.observe(source, { box: 'border-box' });
|
2021-11-05 17:20:06 +08:00
|
|
|
}
|
2022-02-24 02:26:52 +08:00
|
|
|
},
|
|
|
|
|
2022-09-23 21:01:55 +08:00
|
|
|
populateMessageBody = (oMessage, popup) => {
|
2022-02-24 02:26:52 +08:00
|
|
|
if (oMessage) {
|
2022-10-28 19:57:13 +08:00
|
|
|
popup || MessageUserStore.message(oMessage);
|
2022-09-23 21:01:55 +08:00
|
|
|
popup || MessageUserStore.loading(true);
|
2022-02-24 06:11:12 +08:00
|
|
|
Remote.message((iError, oData/*, bCached*/) => {
|
2022-02-24 02:26:52 +08:00
|
|
|
if (iError) {
|
2022-09-23 21:01:55 +08:00
|
|
|
if (Notification.RequestAborted !== iError && !popup) {
|
2022-02-24 02:26:52 +08:00
|
|
|
MessageUserStore.message(null);
|
|
|
|
MessageUserStore.error(getNotification(iError));
|
|
|
|
}
|
|
|
|
} else {
|
2022-09-23 21:01:55 +08:00
|
|
|
let json = oData?.Result;
|
2022-02-24 02:26:52 +08:00
|
|
|
|
|
|
|
if (
|
|
|
|
json &&
|
|
|
|
MessageModel.validJson(json) &&
|
2023-01-25 01:58:25 +08:00
|
|
|
oMessage.folder === json.folder
|
2022-02-24 02:26:52 +08:00
|
|
|
) {
|
2022-09-23 21:01:55 +08:00
|
|
|
const threads = oMessage.threads(),
|
2023-01-25 01:58:25 +08:00
|
|
|
isNew = !popup && oMessage.uid != json.uid && threads.includes(json.uid),
|
2022-02-24 02:26:52 +08:00
|
|
|
messagesDom = MessageUserStore.bodiesDom();
|
2022-09-23 21:01:55 +08:00
|
|
|
if (isNew) {
|
|
|
|
oMessage = MessageModel.reviveFromJson(json);
|
|
|
|
if (oMessage) {
|
|
|
|
oMessage.threads(threads);
|
|
|
|
MessageFlagsCache.initMessage(oMessage);
|
2022-02-24 02:26:52 +08:00
|
|
|
|
|
|
|
// Set clone
|
2022-10-28 19:57:13 +08:00
|
|
|
oMessage = MessageModel.fromMessageListItem(oMessage);
|
2022-02-24 02:26:52 +08:00
|
|
|
}
|
2022-10-28 19:57:13 +08:00
|
|
|
MessageUserStore.message(oMessage);
|
2022-02-24 02:26:52 +08:00
|
|
|
}
|
|
|
|
|
2023-01-25 01:58:25 +08:00
|
|
|
if (oMessage && oMessage.uid == json.uid) {
|
2022-09-23 21:01:55 +08:00
|
|
|
popup || MessageUserStore.error('');
|
2022-02-24 02:26:52 +08:00
|
|
|
/*
|
|
|
|
if (bCached) {
|
2023-01-25 01:58:25 +08:00
|
|
|
delete json.flags;
|
2022-02-24 02:26:52 +08:00
|
|
|
}
|
|
|
|
*/
|
2022-09-23 21:01:55 +08:00
|
|
|
isNew || oMessage.revivePropertiesFromJson(json);
|
2022-10-28 19:57:13 +08:00
|
|
|
|
2022-02-24 02:26:52 +08:00
|
|
|
if (messagesDom) {
|
2022-09-23 21:01:55 +08:00
|
|
|
let id = 'rl-msg-' + oMessage.hash.replace(/[^a-zA-Z0-9]/g, ''),
|
2022-02-24 02:26:52 +08:00
|
|
|
body = elementById(id);
|
|
|
|
if (body) {
|
2022-09-23 21:01:55 +08:00
|
|
|
oMessage.body = body;
|
|
|
|
oMessage.isHtml(body.classList.contains('html'));
|
|
|
|
oMessage.hasImages(body.rlHasImages);
|
2022-02-24 02:26:52 +08:00
|
|
|
} else {
|
|
|
|
body = Element.fromHTML('<div id="' + id + '" hidden="" class="b-text-part '
|
2022-09-23 21:01:55 +08:00
|
|
|
+ (oMessage.pgpSigned() ? ' openpgp-signed' : '')
|
|
|
|
+ (oMessage.pgpEncrypted() ? ' openpgp-encrypted' : '')
|
2022-02-24 02:26:52 +08:00
|
|
|
+ '">'
|
|
|
|
+ '</div>');
|
2022-09-23 21:01:55 +08:00
|
|
|
oMessage.body = body;
|
|
|
|
if (!SettingsUserStore.viewHTML() || !oMessage.viewHtml()) {
|
|
|
|
oMessage.viewPlain();
|
2022-02-24 02:26:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
MessageUserStore.purgeMessageBodyCache();
|
|
|
|
}
|
|
|
|
|
|
|
|
messagesDom.append(body);
|
|
|
|
|
2022-10-28 19:57:13 +08:00
|
|
|
popup || (oMessage.body.hidden = false);
|
2022-09-23 21:01:55 +08:00
|
|
|
popup && oMessage.viewPopupMessage();
|
2022-02-24 02:26:52 +08:00
|
|
|
}
|
|
|
|
|
2022-09-23 21:01:55 +08:00
|
|
|
MessageFlagsCache.initMessage(oMessage);
|
|
|
|
if (oMessage.isUnseen()) {
|
2022-02-24 02:26:52 +08:00
|
|
|
MessageUserStore.MessageSeenTimer = setTimeout(
|
2022-09-23 21:01:55 +08:00
|
|
|
() => MessagelistUserStore.setAction(oMessage.folder, MessageSetAction.SetSeen, [oMessage]),
|
2022-02-24 02:26:52 +08:00
|
|
|
SettingsUserStore.messageReadDelay() * 1000 // seconds
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-09-23 21:01:55 +08:00
|
|
|
if (isNew) {
|
2022-02-24 02:26:52 +08:00
|
|
|
let selectedMessage = MessagelistUserStore.selectedMessage();
|
|
|
|
if (
|
|
|
|
selectedMessage &&
|
2022-09-23 21:01:55 +08:00
|
|
|
(oMessage.folder !== selectedMessage.folder || oMessage.uid != selectedMessage.uid)
|
2022-02-24 02:26:52 +08:00
|
|
|
) {
|
|
|
|
MessagelistUserStore.selectedMessage(null);
|
|
|
|
if (1 === MessagelistUserStore.length) {
|
|
|
|
MessagelistUserStore.focusedMessage(null);
|
|
|
|
}
|
|
|
|
} else if (!selectedMessage) {
|
|
|
|
selectedMessage = MessagelistUserStore.find(
|
|
|
|
subMessage =>
|
|
|
|
subMessage &&
|
2022-09-23 21:01:55 +08:00
|
|
|
subMessage.folder === oMessage.folder &&
|
|
|
|
subMessage.uid == oMessage.uid
|
2022-02-24 02:26:52 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
if (selectedMessage) {
|
|
|
|
MessagelistUserStore.selectedMessage(selectedMessage);
|
|
|
|
MessagelistUserStore.focusedMessage(selectedMessage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-09-23 21:01:55 +08:00
|
|
|
popup || MessageUserStore.loading(false);
|
2022-02-24 02:26:52 +08:00
|
|
|
}, oMessage.folder, oMessage.uid);
|
|
|
|
}
|
2021-11-05 17:20:06 +08:00
|
|
|
};
|
2022-11-22 18:14:56 +08:00
|
|
|
|
|
|
|
leftPanelDisabled.subscribe(value => value && moveAction(false));
|
|
|
|
moveAction.subscribe(value => value && leftPanelDisabled(false));
|