import ko from 'ko'; import { MessagePriority } from 'Common/EnumsUser'; import { i18n } from 'Common/Translator'; import { encodeHtml } from 'Common/Html'; import { isArray, arrayLength } from 'Common/Utils'; import { serverRequestRaw } from 'Common/Links'; import { FolderUserStore } from 'Stores/User/Folder'; import { FileInfo } from 'Common/File'; import { AttachmentCollectionModel } from 'Model/AttachmentCollection'; import { EmailCollectionModel } from 'Model/EmailCollection'; import { AbstractModel } from 'Knoin/AbstractModel'; import PreviewHTML from 'Html/PreviewMessage.html'; const SignedVerifyStatus = { UnknownPublicKeys: -4, UnknownPrivateKey: -3, Unverified: -2, Error: -1, None: 0, Success: 1 }, replyHelper = (emails, unic, localEmails) => { emails.forEach(email => { if (undefined === unic[email.email]) { unic[email.email] = true; localEmails.push(email); } }); }; export class MessageModel extends AbstractModel { constructor() { super(); this._reset(); this.addObservables({ subject: '', size: 0, spamScore: 0, spamResult: '', isSpam: false, hasVirus: null, // or boolean when scanned dateTimeStampInUTC: 0, priority: MessagePriority.Normal, senderEmailsString: '', senderClearEmailsString: '', deleted: false, isDeleted: false, isUnseen: false, isFlagged: false, isAnswered: false, isForwarded: false, isReadReceipt: false, focused: false, selected: false, checked: false, hasAttachments: false, isHtml: false, hasImages: false, isPgpSigned: false, isPgpEncrypted: false, pgpSignedVerifyStatus: SignedVerifyStatus.None, pgpSignedVerifyUser: '', readReceipt: '', hasUnseenSubMessage: false, hasFlaggedSubMessage: false }); this.attachments = ko.observableArray(new AttachmentCollectionModel); this.attachmentsSpecData = ko.observableArray(); this.threads = ko.observableArray(); this.unsubsribeLinks = ko.observableArray(); this.addComputables({ attachmentIconClass: () => FileInfo.getCombinedIconClass(this.hasAttachments() ? this.attachmentsSpecData() : []), threadsLen: () => this.threads().length, isImportant: () => MessagePriority.High === this.priority(), }); } _reset() { this.folder = ''; this.uid = 0; this.hash = ''; this.requestHash = ''; this.externalProxy = false; this.emails = []; this.from = new EmailCollectionModel; this.to = new EmailCollectionModel; this.cc = new EmailCollectionModel; this.bcc = new EmailCollectionModel; this.replyTo = new EmailCollectionModel; this.deliveredTo = new EmailCollectionModel; this.body = null; this.draftInfo = []; this.messageId = ''; this.inReplyTo = ''; this.references = ''; } clear() { this._reset(); this.subject(''); this.size(0); this.spamScore(0); this.spamResult(''); this.isSpam(false); this.hasVirus(null); this.dateTimeStampInUTC(0); this.priority(MessagePriority.Normal); this.senderEmailsString(''); this.senderClearEmailsString(''); this.deleted(false); this.isDeleted(false); this.isUnseen(false); this.isFlagged(false); this.isAnswered(false); this.isForwarded(false); this.isReadReceipt(false); this.selected(false); this.checked(false); this.hasAttachments(false); this.attachmentsSpecData([]); this.isHtml(false); this.hasImages(false); this.attachments(new AttachmentCollectionModel); this.isPgpSigned(false); this.isPgpEncrypted(false); this.pgpSignedVerifyStatus(SignedVerifyStatus.None); this.pgpSignedVerifyUser(''); this.priority(MessagePriority.Normal); this.readReceipt(''); this.threads([]); this.unsubsribeLinks([]); this.hasUnseenSubMessage(false); this.hasFlaggedSubMessage(false); } /** * @param {Array} properties * @returns {Array} */ getEmails(properties) { return properties.reduce((carry, property) => carry.concat(this[property]), []).map( oItem => oItem ? oItem.email : '' ).validUnique(); } /** * @returns {string} */ friendlySize() { return FileInfo.friendlySize(this.size()); } computeSenderEmail() { const list = [FolderUserStore.sentFolder(), FolderUserStore.draftFolder()].includes(this.folder) ? 'to' : 'from'; this.senderEmailsString(this[list].toString(true)); this.senderClearEmailsString(this[list].toStringClear()); } /** * @param {FetchJsonMessage} json * @returns {boolean} */ revivePropertiesFromJson(json) { if ('Priority' in json && ![MessagePriority.High, MessagePriority.Low].includes(json.Priority)) { json.Priority = MessagePriority.Normal; } if (super.revivePropertiesFromJson(json)) { // this.foundedCIDs = isArray(json.FoundedCIDs) ? json.FoundedCIDs : []; // this.attachments(AttachmentCollectionModel.reviveFromJson(json.Attachments, this.foundedCIDs)); this.computeSenderEmail(); } } /** * @returns {boolean} */ hasUnsubsribeLinks() { return this.unsubsribeLinks().length; } /** * @returns {string} */ getFirstUnsubsribeLink() { return this.unsubsribeLinks()[0] || ''; } /** * @param {boolean} friendlyView * @param {boolean=} wrapWithLink * @returns {string} */ fromToLine(friendlyView, wrapWithLink) { return this.from.toString(friendlyView, wrapWithLink); } /** * @returns {string} */ fromDkimData() { let result = ['none', '']; if (1 === arrayLength(this.from) && this.from[0] && this.from[0].dkimStatus) { result = [this.from[0].dkimStatus, this.from[0].dkimValue || '']; } return result; } /** * @param {boolean} friendlyView * @param {boolean=} wrapWithLink * @returns {string} */ toToLine(friendlyView, wrapWithLink) { return this.to.toString(friendlyView, wrapWithLink); } /** * @param {boolean} friendlyView * @param {boolean=} wrapWithLink * @returns {string} */ ccToLine(friendlyView, wrapWithLink) { return this.cc.toString(friendlyView, wrapWithLink); } /** * @param {boolean} friendlyView * @param {boolean=} wrapWithLink * @returns {string} */ bccToLine(friendlyView, wrapWithLink) { return this.bcc.toString(friendlyView, wrapWithLink); } /** * @param {boolean} friendlyView * @param {boolean=} wrapWithLink * @returns {string} */ replyToToLine(friendlyView, wrapWithLink) { return this.replyTo.toString(friendlyView, wrapWithLink); } /** * @return string */ lineAsCss() { let classes = []; Object.entries({ deleted: this.deleted(), 'deleted-mark': this.isDeleted(), selected: this.selected(), checked: this.checked(), flagged: this.isFlagged(), unseen: this.isUnseen(), answered: this.isAnswered(), forwarded: this.isForwarded(), focused: this.focused(), important: this.isImportant(), withAttachments: this.hasAttachments(), emptySubject: !this.subject(), // hasChildrenMessage: 1 < this.threadsLen(), hasUnseenSubMessage: this.hasUnseenSubMessage(), hasFlaggedSubMessage: this.hasFlaggedSubMessage() }).forEach(([key, value]) => value && classes.push(key)); return classes.join(' '); } /** * @returns {string} */ fromAsSingleEmail() { return isArray(this.from) && this.from[0] ? this.from[0].email : ''; } /** * @returns {string} */ viewLink() { return serverRequestRaw('ViewAsPlain', this.requestHash); } /** * @returns {string} */ downloadLink() { return serverRequestRaw('Download', this.requestHash); } /** * @param {Object} excludeEmails * @param {boolean=} last = false * @returns {Array} */ replyEmails(excludeEmails, last) { const result = [], unic = undefined === excludeEmails ? {} : excludeEmails; replyHelper(this.replyTo, unic, result); if (!result.length) { replyHelper(this.from, unic, result); } if (!result.length && !last) { return this.replyEmails({}, true); } return result; } /** * @param {Object} excludeEmails * @param {boolean=} last = false * @returns {Array.} */ replyAllEmails(excludeEmails, last) { let data = []; const toResult = [], ccResult = [], unic = undefined === excludeEmails ? {} : excludeEmails; replyHelper(this.replyTo, unic, toResult); if (!toResult.length) { replyHelper(this.from, unic, toResult); } replyHelper(this.to, unic, toResult); replyHelper(this.cc, unic, ccResult); if (!toResult.length && !last) { data = this.replyAllEmails({}, true); return [data[0], ccResult]; } return [toResult, ccResult]; } /** * @param {boolean=} print = false */ viewPopupMessage(print) { const timeStampInUTC = this.dateTimeStampInUTC() || 0, ccLine = this.ccToLine(false), m = 0 < timeStampInUTC ? new Date(timeStampInUTC * 1000) : null, win = open(''), doc = win.document; doc.write(PreviewHTML .replace(/{{subject}}/g, encodeHtml(this.subject())) .replace('{{date}}', encodeHtml(m ? m.format('LLL') : '')) .replace('{{fromCreds}}', encodeHtml(this.fromToLine(false))) .replace('{{toCreds}}', encodeHtml(this.toToLine(false))) .replace('{{toLabel}}', encodeHtml(i18n('GLOBAL/TO'))) .replace('{{ccHide}}', ccLine ? '' : 'hidden=""') .replace('{{ccCreds}}', encodeHtml(ccLine)) .replace('{{ccLabel}}', encodeHtml(i18n('GLOBAL/CC'))) .replace('{{bodyClass}}', this.isHtml() ? 'html' : 'plain') .replace('{{html}}', this.bodyAsHTML()) ); doc.close(); if (print) { setTimeout(() => win.print(), 100); } } printMessage() { this.viewPopupMessage(true); } /** * @returns {string} */ generateUid() { return this.folder + '/' + this.uid; } /** * @param {MessageModel} message * @returns {MessageModel} */ populateByMessageListItem(message) { if (message) { this.folder = message.folder; this.uid = message.uid; this.hash = message.hash; this.requestHash = message.requestHash; this.subject(message.subject()); this.size(message.size()); this.spamScore(message.spamScore()); this.spamResult(message.spamResult()); this.isSpam(message.isSpam()); this.hasVirus(message.hasVirus()); this.dateTimeStampInUTC(message.dateTimeStampInUTC()); this.priority(message.priority()); this.externalProxy = message.externalProxy; this.emails = message.emails; this.from = message.from; this.to = message.to; this.cc = message.cc; this.bcc = message.bcc; this.replyTo = message.replyTo; this.deliveredTo = message.deliveredTo; this.unsubsribeLinks(message.unsubsribeLinks); this.isUnseen(message.isUnseen()); this.isFlagged(message.isFlagged()); this.isAnswered(message.isAnswered()); this.isForwarded(message.isForwarded()); this.isReadReceipt(message.isReadReceipt()); this.isDeleted(message.isDeleted()); this.priority(message.priority()); this.selected(message.selected()); this.checked(message.checked()); this.hasAttachments(message.hasAttachments()); this.attachmentsSpecData(message.attachmentsSpecData()); } this.body = null; this.draftInfo = []; this.messageId = ''; this.inReplyTo = ''; this.references = ''; if (message) { this.threads(message.threads()); } this.computeSenderEmail(); return this; } showExternalImages() { if (this.body && this.body.rlHasImages) { this.hasImages(false); this.body.rlHasImages = false; let body = this.body, attr = this.externalProxy ? 'data-x-additional-src' : 'data-x-src'; body.querySelectorAll('[' + attr + ']').forEach(node => { if (node.matches('img')) { node.loading = 'lazy'; } node.src = node.getAttribute(attr); }); attr = this.externalProxy ? 'data-x-additional-style-url' : 'data-x-style-url'; body.querySelectorAll('[' + attr + ']').forEach(node => { node.setAttribute('style', ((node.getAttribute('style')||'') + ';' + node.getAttribute(attr)) .replace(/^[;\s]+/,'')); }); } } showInternalImages() { const body = this.body; if (body && !body.rlInitInternalImages) { const findAttachmentByCid = cid => this.attachments().findByCid(cid); body.rlInitInternalImages = true; body.querySelectorAll('[data-x-src-cid],[data-x-src-location],[data-x-style-cid]').forEach(el => { const data = el.dataset; if (data.xSrcCid) { const attachment = findAttachmentByCid(data.xSrcCid); if (attachment && attachment.download) { el.src = attachment.linkPreview(); } } else if (data.xSrcLocation) { const attachment = this.attachments.find(item => data.xSrcLocation === item.contentLocation) || findAttachmentByCid(data.xSrcLocation); if (attachment && attachment.download) { el.loading = 'lazy'; el.src = attachment.linkPreview(); } } else if (data.xStyleCid) { const name = data.xStyleCidName, attachment = findAttachmentByCid(data.xStyleCid); if (attachment && attachment.linkPreview && name) { el.setAttribute('style', name + ": url('" + attachment.linkPreview() + "');" + (el.getAttribute('style') || '')); } } }); } } fetchDataFromDom() { if (this.body) { this.isHtml(!!this.body.rlIsHtml); this.hasImages(!!this.body.rlHasImages); } } /** * @returns {string} */ bodyAsHTML() { if (this.body) { let clone = this.body.cloneNode(true), attr = 'data-html-editor-font-wrapper'; clone.querySelectorAll('blockquote.rl-bq-switcher').forEach( node => node.classList.remove('rl-bq-switcher','hidden-bq') ); clone.querySelectorAll('.rlBlockquoteSwitcher').forEach( node => node.remove() ); clone.querySelectorAll('['+attr+']').forEach( node => node.removeAttribute(attr) ); return clone.innerHTML; } return ''; } /** * @returns {string} */ flagHash() { return [ this.deleted(), this.isDeleted(), this.isUnseen(), this.isFlagged(), this.isAnswered(), this.isForwarded(), this.isReadReceipt() ].join(','); } }