/** * @module helpers/html */ import { Dom } from 'jodit/core/dom/dom'; import { $$ } from 'jodit/core/helpers/utils'; import { trim } from '../string/trim'; function normalizeCSS(s: string): string { return s .replace(/mso-[a-z-]+:[\s]*[^;]+;/gi, '') .replace(/mso-[a-z-]+:[\s]*[^";']+$/gi, '') .replace(/border[a-z-]*:[\s]*[^;]+;/gi, '') .replace(/([0-9.]+)(pt|cm)/gi, (match, units, metrics) => { switch (metrics.toLowerCase()) { case 'pt': return (parseFloat(units) * 1.328).toFixed(0) + 'px'; case 'cm': return (parseFloat(units) * 0.02645833).toFixed(0) + 'px'; } return match; }); } /** * If the HTML has CSS rules with selectors, * it applies them to the selectors in the HTML itself * and then removes the selector styles, leaving only the inline ones. */ export function applyStyles(html: string): string { if (html.indexOf('') + ''.length); const iframe = document.createElement('iframe'); iframe.style.display = 'none'; document.body.appendChild(iframe); let convertedString: string = '', collection: HTMLElement[] = []; try { const iframeDoc: Document | null = iframe.contentDocument || (iframe.contentWindow ? iframe.contentWindow.document : null); if (iframeDoc) { iframeDoc.open(); iframeDoc.write(html); iframeDoc.close(); try { for (let i = 0; i < iframeDoc.styleSheets.length; i += 1) { const rules: CSSStyleRule[] = ( iframeDoc.styleSheets[i] as any ).cssRules; for (let idx = 0; idx < rules.length; idx += 1) { if (rules[idx].selectorText === '') { continue; } collection = $$( rules[idx].selectorText, iframeDoc.body ); collection.forEach((elm: HTMLElement) => { elm.style.cssText = normalizeCSS( rules[idx].style.cssText + ';' + elm.style.cssText ); }); } } } catch (e) { if (!isProd) { throw e; } } Dom.each(iframeDoc.body, node => { if (Dom.isElement(node)) { const elm = node as HTMLElement; const css = elm.getAttribute('style'); if (css) { elm.style.cssText = normalizeCSS(css); } if ( elm.hasAttribute('style') && !elm.getAttribute('style') ) { elm.removeAttribute('style'); } } }); convertedString = iframeDoc.firstChild ? trim(iframeDoc.body.innerHTML) : ''; } } catch { } finally { Dom.safeRemove(iframe); } if (convertedString) { html = convertedString; } return trim( html .replace(/<(\/)?(html|colgroup|col|o:p)[^>]*>/g, '') .replace(/