import ko from 'ko'; import { MessagePriority } from 'Common/EnumsUser'; import { i18n } from 'Common/Translator'; import { doc, SettingsGet } from 'Common/Globals'; import { encodeHtml, plainToHtml, htmlToPlain, cleanHtml } from 'Common/Html'; import { arrayLength, forEachObjectEntry } from 'Common/Utils'; import { serverRequestRaw, proxy } from 'Common/Links'; import { addObservablesTo, addComputablesTo } from 'External/ko'; import { FolderUserStore, isAllowedKeyword } from 'Stores/User/Folder'; import { SettingsUserStore } from 'Stores/User/Settings'; 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'; import { LanguageStore } from 'Stores/Language'; //import { MessageFlagsCache } from 'Common/Cache'; import Remote from 'Remote/User/Fetch'; const hcont = Element.fromHTML('
'), getRealHeight = el => { hcont.innerHTML = el.outerHTML; const result = hcont.clientHeight; hcont.innerHTML = ''; return result; }, toggleTag = (message, keyword) => { const lower = keyword.toLowerCase(), flags = message.flags, isSet = flags.includes(lower); Remote.request('MessageSetKeyword', iError => { if (!iError) { isSet ? flags.remove(lower) : flags.push(lower); // MessageFlagsCache.setFor(message.folder, message.uid, flags()); } }, { Folder: message.folder, Uids: message.uid, Keyword: keyword, SetAction: isSet ? 0 : 1 }) }, /** * @param {EmailCollectionModel} emails * @param {Object} unic * @param {Map} localEmails */ replyHelper = (emails, unic, localEmails) => emails.forEach(email => unic[email.email] || localEmails.has(email.email) || localEmails.set(email.email, email) ); doc.body.append(hcont); export class MessageModel extends AbstractModel { constructor() { super(); this._reset(); addObservablesTo(this, { subject: '', plain: '', html: '', size: 0, spamScore: 0, spamResult: '', isSpam: false, hasVirus: null, // or boolean when scanned dateTimeStampInUTC: 0, priority: MessagePriority.Normal, senderEmailsString: '', senderClearEmailsString: '', deleted: false, // Also used by Selector focused: false, selected: false, checked: false, isHtml: false, hasImages: false, hasExternals: false, pgpSigned: null, pgpVerified: null, encrypted: false, pgpEncrypted: null, pgpDecrypted: false, readReceipt: '', hasUnseenSubMessage: false, hasFlaggedSubMessage: false }); this.attachments = ko.observableArray(new AttachmentCollectionModel); this.threads = ko.observableArray(); this.unsubsribeLinks = ko.observableArray(); this.flags = ko.observableArray(); addComputablesTo(this, { attachmentIconClass: () => this.encrypted() ? 'icon-lock' : FileInfo.getAttachmentsIconClass(this.attachments()), threadsLen: () => this.threads().length, listAttachments: () => this.attachments() .filter(item => SettingsUserStore.listInlineAttachments() || !item.isLinked()), hasAttachments: () => this.listAttachments().length, isUnseen: () => !this.flags().includes('\\seen'), isFlagged: () => this.flags().includes('\\flagged'), isReadReceipt: () => this.flags().includes('$mdnsent'), // isJunk: () => this.flags().includes('$junk') && !this.flags().includes('$nonjunk'), // isPhishing: () => this.flags().includes('$phishing'), tagsToHTML: () => this.flags().map(value => isAllowedKeyword(value) ? '' + i18n('MESSAGE_TAGS/'+value,0,value) + '' : '' ).join(' '), tagOptions: () => { const tagOptions = []; FolderUserStore.currentFolder().permanentFlags.forEach(value => { if (isAllowedKeyword(value)) { let lower = value.toLowerCase(); tagOptions.push({ css: 'msgflag-' + lower, value: value, checked: this.flags().includes(lower), label: i18n('MESSAGE_TAGS/'+lower, 0, value), toggle: (/*obj*/) => toggleTag(this, value) }); } }); return tagOptions } }); } toggleTag(keyword) { toggleTag(this, keyword); } _reset() { this.folder = ''; this.uid = 0; this.hash = ''; this.requestHash = ''; 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.html(''); this.plain(''); 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.selected(false); this.checked(false); this.isHtml(false); this.hasImages(false); this.hasExternals(false); this.attachments(new AttachmentCollectionModel); this.pgpSigned(null); this.pgpVerified(null); this.pgpEncrypted(null); this.pgpDecrypted(false); this.priority(MessagePriority.Normal); this.readReceipt(''); this.threads([]); this.unsubsribeLinks([]); this.hasUnseenSubMessage(false); this.hasFlaggedSubMessage(false); } spamStatus() { let spam = this.spamResult(); return spam ? i18n(this.isSpam() ? 'GLOBAL/SPAM' : 'GLOBAL/NOT_SPAM') + ': ' + spam : ''; } /** * @returns {string} */ friendlySize() { return FileInfo.friendlySize(this.size()); } computeSenderEmail() { const list = this[ [FolderUserStore.sentFolder(), FolderUserStore.draftsFolder()].includes(this.folder) ? 'to' : 'from' ]; this.senderEmailsString(list.toString(true)); this.senderClearEmailsString(list.map(email => email?.email).filter(email => email).join(', ')); } /** * @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.foundCIDs = isArray(json.FoundCIDs) ? json.FoundCIDs : []; // this.attachments(AttachmentCollectionModel.reviveFromJson(json.Attachments, this.foundCIDs)); 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]?.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); } /** * @returns {string} */ bccToLine() { return this.bcc.toString(); } /** * @returns {string} */ replyToToLine() { return this.replyTo.toString(); } /** * @return string */ lineAsCss(flags=1) { let classes = []; forEachObjectEntry({ deleted: this.deleted(), selected: this.selected(), checked: this.checked(), unseen: this.isUnseen(), focused: this.focused(), priorityHigh: this.priority() === MessagePriority.High, withAttachments: !!this.attachments().length, // hasChildrenMessage: 1 < this.threadsLen(), hasUnseenSubMessage: this.hasUnseenSubMessage(), hasFlaggedSubMessage: this.hasFlaggedSubMessage() }, (key, value) => value && classes.push(key)); flags && this.flags().forEach(value => classes.push('msgflag-'+value)); return classes.join(' '); } /** * @returns {string} */ viewLink() { return serverRequestRaw('ViewAsPlain', this.requestHash); } /** * @returns {string} */ downloadLink() { return serverRequestRaw('Download', this.requestHash); } /** * @param {Object} excludeEmails * @returns {Array} */ replyEmails(excludeEmails) { const result = new Map(), unic = excludeEmails || {}; replyHelper(this.replyTo, unic, result); result.size || replyHelper(this.from, unic, result); return result.size ? [...result.values()] : [this.to[0]]; } /** * @param {Object} excludeEmails * @returns {Array.