import { ComposeType, FolderType } from 'Common/EnumsUser'; import { EmailModel } from 'Model/Email'; import { encodeHtml } from 'Common/Html'; import { isArray } from 'Common/Utils'; import { createElement } from 'Common/Globals'; /** * @param {(string|number)} value * @param {boolean=} includeZero = true * @returns {boolean} */ export function isPosNumeric(value, includeZero = true) { return null != value && (includeZero ? /^[0-9]*$/ : /^[1-9]+[0-9]*$/).test(value.toString()); } /** * @param {string} text * @param {number=} len = 100 * @returns {string} */ function splitPlainText(text, len = 100) { let prefix = '', subText = '', result = text, spacePos = 0, newLinePos = 0; while (result.length > len) { subText = result.substr(0, len); spacePos = subText.lastIndexOf(' '); newLinePos = subText.lastIndexOf('\n'); if (-1 !== newLinePos) { spacePos = newLinePos; } if (-1 === spacePos) { spacePos = len; } prefix += subText.substr(0, spacePos) + '\n'; result = result.substr(spacePos + 1); } return prefix + result; } /** * @param {string} html * @returns {string} */ export function htmlToPlain(html) { let pos = 0, limit = 0, iP1 = 0, iP2 = 0, iP3 = 0, text = ''; const convertBlockquote = (blockquoteText) => { blockquoteText = '> ' + blockquoteText.trim().replace(/\n/gm, '\n> '); return blockquoteText.replace(/(^|\n)([> ]+)/gm, (...args) => args && 2 < args.length ? args[1] + args[2].replace(/[\s]/g, '').trim() + ' ' : '' ); }; const convertDivs = (...args) => { if (args && 1 < args.length) { let divText = args[1].trim(); if (divText.length) { divText = divText.replace(/]*>([\s\S\r\n]*)<\/div>/gim, convertDivs); divText = '\n' + divText.trim() + '\n'; } return divText; } return ''; }; const tpl = createElement('template'), convertPre = (...args) => args && 1 < args.length ? args[1] .toString() .replace(/[\n]/gm, '
') .replace(/[\r]/gm, '') : '', fixAttibuteValue = (...args) => (args && 1 < args.length ? '' + args[1] + encodeHtml(args[2]) : ''), convertLinks = (...args) => (args && 1 < args.length ? args[1].trim() : ''); tpl.innerHTML = html .replace(/]*><\/p>/gi, '') .replace(/]*>([\s\S\r\n\t]*)<\/pre>/gim, convertPre) .replace(/[\s]+/gm, ' ') .replace(/((?:href|data)\s?=\s?)("[^"]+?"|'[^']+?')/gim, fixAttibuteValue) .replace(/]*>/gim, '\n') .replace(/<\/h[\d]>/gi, '\n') .replace(/<\/p>/gi, '\n\n') .replace(/]*>/gim, '\n') .replace(/<\/ul>/gi, '\n') .replace(/]*>/gim, ' * ') .replace(/<\/li>/gi, '\n') .replace(/<\/td>/gi, '\n') .replace(/<\/tr>/gi, '\n') .replace(/]*>/gim, '\n_______________________________\n\n') .replace(/]*>([\s\S\r\n]*)<\/div>/gim, convertDivs) .replace(/]*>/gim, '\n__bq__start__\n') .replace(/<\/blockquote>/gim, '\n__bq__end__\n') .replace(/]*>([\s\S\r\n]*?)<\/a>/gim, convertLinks) .replace(/<\/div>/gi, '\n') .replace(/ /gi, ' ') .replace(/"/gi, '"') .replace(/<[^>]*>/gm, ''); text = splitPlainText(tpl.content.textContent .replace(/\n[ \t]+/gm, '\n') .replace(/[\n]{3,}/gm, '\n\n') .replace(/>/gi, '>') .replace(/</gi, '<') .replace(/&/gi, '&') ); pos = 0; limit = 800; while (0 < limit) { --limit; iP1 = text.indexOf('__bq__start__', pos); if (-1 < iP1) { iP2 = text.indexOf('__bq__start__', iP1 + 5); iP3 = text.indexOf('__bq__end__', iP1 + 5); if ((-1 === iP2 || iP3 < iP2) && iP1 < iP3) { text = text.substr(0, iP1) + convertBlockquote(text.substring(iP1 + 13, iP3)) + text.substr(iP3 + 11); pos = 0; } else if (-1 < iP2 && iP2 < iP3) { pos = iP2 - 1; } else { pos = 0; } } else { break; } } return text.replace(/__bq__start__|__bq__end__/gm, '').trim(); } /** * @param {string} plain * @param {boolean} findEmailAndLinksInText = false * @returns {string} */ export function plainToHtml(plain) { plain = plain.toString().replace(/\r/g, ''); plain = plain.replace(/^>[> ]>+/gm, ([match]) => (match ? match.replace(/[ ]+/g, '') : match)); let bIn = false, bDo = true, bStart = true, aNextText = [], aText = plain.split('\n'); do { bDo = false; aNextText = []; aText.forEach(sLine => { bStart = '>' === sLine.substr(0, 1); if (bStart && !bIn) { bDo = true; bIn = true; aNextText.push('~~~blockquote~~~'); aNextText.push(sLine.substr(1)); } else if (!bStart && bIn) { if (sLine) { bIn = false; aNextText.push('~~~/blockquote~~~'); aNextText.push(sLine); } else { aNextText.push(sLine); } } else if (bStart && bIn) { aNextText.push(sLine.substr(1)); } else { aNextText.push(sLine); } }); if (bIn) { bIn = false; aNextText.push('~~~/blockquote~~~'); } aText = aNextText; } while (bDo); return aText.join('\n') // .replace(/~~~\/blockquote~~~\n~~~blockquote~~~/g, '\n') .replace(/&/g, '&') .replace(/>/g, '>') .replace(/') .replace(/[\s]*~~~\/blockquote~~~/g, '') .replace(/\n/g, '
'); } rl.Utils = { htmlToPlain: htmlToPlain, plainToHtml: plainToHtml }; /** * @param {Array} aSystem * @param {Array} aList * @param {Array=} aDisabled * @param {Array=} aHeaderLines * @param {Function=} fDisableCallback * @param {Function=} fRenameCallback * @param {boolean=} bSystem * @param {boolean=} bBuildUnvisible * @returns {Array} */ export function folderListOptionsBuilder( aSystem, aList, aDisabled, aHeaderLines, fDisableCallback, fRenameCallback, bSystem, bBuildUnvisible ) { let /** * @type {?FolderModel} */ bSep = false, aResult = []; const sDeepPrefix = '\u00A0\u00A0\u00A0'; bBuildUnvisible = undefined === bBuildUnvisible ? false : !!bBuildUnvisible; bSystem = null == bSystem ? 0 < aSystem.length : bSystem; fDisableCallback = null != fDisableCallback ? fDisableCallback : null; fRenameCallback = null != fRenameCallback ? fRenameCallback : null; if (!isArray(aDisabled)) { aDisabled = []; } if (!isArray(aHeaderLines)) { aHeaderLines = []; } aHeaderLines.forEach(line => { aResult.push({ id: line[0], name: line[1], system: false, dividerbar: false, disabled: false }); }); bSep = true; aSystem.forEach(oItem => { aResult.push({ id: oItem.fullNameRaw, name: fRenameCallback ? fRenameCallback(oItem) : oItem.name(), system: true, dividerbar: bSep, disabled: !oItem.selectable || aDisabled.includes(oItem.fullNameRaw) || (fDisableCallback ? fDisableCallback(oItem) : false) }); bSep = false; }); bSep = true; aList.forEach(oItem => { // if (oItem.subscribed() || !oItem.exists || bBuildUnvisible) if ( (oItem.subscribed() || !oItem.exists || bBuildUnvisible) && (oItem.selectable || oItem.hasSubscribedSubfolders()) ) { if (FolderType.User === oItem.type() || !bSystem || oItem.hasSubscribedSubfolders()) { aResult.push({ id: oItem.fullNameRaw, name: sDeepPrefix.repeat(oItem.deep + 1) + (fRenameCallback ? fRenameCallback(oItem) : oItem.name()), system: false, dividerbar: bSep, disabled: !oItem.selectable || aDisabled.includes(oItem.fullNameRaw) || (fDisableCallback ? fDisableCallback(oItem) : false) }); bSep = false; } } if (oItem.subscribed() && oItem.subFolders.length) { aResult = aResult.concat( folderListOptionsBuilder( [], oItem.subFolders(), aDisabled, [], fDisableCallback, fRenameCallback, bSystem, bBuildUnvisible ) ); } }); return aResult; } /** * Call the Model/CollectionModel onDestroy() to clear knockout functions/objects * @param {Object|Array} objectOrObjects * @returns {void} */ export function delegateRunOnDestroy(objectOrObjects) { objectOrObjects && (isArray(objectOrObjects) ? objectOrObjects : [objectOrObjects]).forEach( obj => obj.onDestroy && obj.onDestroy() ); } /** * @returns {function} */ export function computedPaginatorHelper(koCurrentPage, koPageCount) { 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() }; if (push) { result.push(data); } else { result.unshift(data); } }; 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) { fAdd(Math.round((prev - 1) / 2), false, '...'); } if (pageCount - 2 === next) { fAdd(pageCount - 1, true); } else if (pageCount - 2 > next) { fAdd(Math.round((pageCount + next) / 2), true, '...'); } // first and last if (1 < prev) { fAdd(1, false); } if (pageCount > next) { fAdd(pageCount, true); } } return result; }; } /** * @param {string} mailToUrl * @returns {boolean} */ export function mailToHelper(mailToUrl) { if ( mailToUrl && 'mailto:' === mailToUrl .toString() .substr(0, 7) .toLowerCase() ) { mailToUrl = mailToUrl.toString().substr(7); let to = [], cc = null, bcc = null, params = {}; const email = mailToUrl.replace(/\?.+$/, ''), query = mailToUrl.replace(/^[^?]*\?/, ''); query.split('&').forEach(temp => { temp = temp.split('='); params[decodeURIComponent(temp[0])] = decodeURIComponent(temp[1]); }); if (undefined !== params.to) { to = EmailModel.parseEmailLine(decodeURIComponent(email + ',' + params.to)); to = Object.values( 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; }, {}) ); } else { to = EmailModel.parseEmailLine(email); } if (undefined !== params.cc) { cc = EmailModel.parseEmailLine(decodeURIComponent(params.cc)); } if (undefined !== params.bcc) { bcc = EmailModel.parseEmailLine(decodeURIComponent(params.bcc)); } showMessageComposer([ ComposeType.Empty, null, to, cc, bcc, null == params.subject ? null : decodeURIComponent(params.subject), null == params.body ? null : plainToHtml(decodeURIComponent(params.body)) ]); return true; } return false; } export function showMessageComposer(params = []) { rl.app.showMessageComposer(params); }