/** * Toc.js * * Released under LGPL License. * Copyright (c) 1999-2017 Ephox Corp. All rights reserved * * License: http://www.tinymce.com/license * Contributing: http://www.tinymce.com/contributing */ import DOMUtils from 'tinymce/core/api/dom/DOMUtils'; import I18n from 'tinymce/core/api/util/I18n'; import Tools from 'tinymce/core/api/util/Tools'; import Settings from '../api/Settings'; import Guid from './Guid'; const tocId = Guid.create('mcetoc_'); const generateSelector = function generateSelector(depth) { let i; const selector = []; for (i = 1; i <= depth; i++) { selector.push('h' + i); } return selector.join(','); }; const hasHeaders = function (editor) { return readHeaders(editor).length > 0; }; const readHeaders = function (editor) { const tocClass = Settings.getTocClass(editor); const headerTag = Settings.getTocHeader(editor); const selector = generateSelector(Settings.getTocDepth(editor)); let headers = editor.$(selector); // if headerTag is one of h1-9, we need to filter it out from the set if (headers.length && /^h[1-9]$/i.test(headerTag)) { headers = headers.filter(function (i, el) { return !editor.dom.hasClass(el.parentNode, tocClass); }); } return Tools.map(headers, function (h) { return { id: h.id ? h.id : tocId(), level: parseInt(h.nodeName.replace(/^H/i, ''), 10), title: editor.$.text(h), element: h }; }); }; const getMinLevel = function (headers) { let i, minLevel = 9; for (i = 0; i < headers.length; i++) { if (headers[i].level < minLevel) { minLevel = headers[i].level; } // do not proceed if we have reached absolute minimum if (minLevel === 1) { return minLevel; } } return minLevel; }; const generateTitle = function (tag, title) { const openTag = '<' + tag + ' contenteditable="true">'; const closeTag = ''; return openTag + DOMUtils.DOM.encode(title) + closeTag; }; const generateTocHtml = function (editor) { const html = generateTocContentHtml(editor); return '
' + html + '
'; }; const generateTocContentHtml = function (editor) { let html = ''; const headers = readHeaders(editor); let prevLevel = getMinLevel(headers) - 1; let i, ii, h, nextLevel; if (!headers.length) { return ''; } html += generateTitle(Settings.getTocHeader(editor), I18n.translate('Table of Contents')); for (i = 0; i < headers.length; i++) { h = headers[i]; h.element.id = h.id; nextLevel = headers[i + 1] && headers[i + 1].level; if (prevLevel === h.level) { html += '
  • '; } else { for (ii = prevLevel; ii < h.level; ii++) { html += ''; } } else { for (ii = h.level; ii > nextLevel; ii--) { html += '
  • '; } } prevLevel = h.level; } return html; }; const isEmptyOrOffscren = function (editor, nodes) { return !nodes.length || editor.dom.getParents(nodes[0], '.mce-offscreen-selection').length > 0; }; const insertToc = function (editor) { const tocClass = Settings.getTocClass(editor); const $tocElm = editor.$('.' + tocClass); if (isEmptyOrOffscren(editor, $tocElm)) { editor.insertContent(generateTocHtml(editor)); } else { updateToc(editor); } }; const updateToc = function (editor) { const tocClass = Settings.getTocClass(editor); const $tocElm = editor.$('.' + tocClass); if ($tocElm.length) { editor.undoManager.transact(function () { $tocElm.html(generateTocContentHtml(editor)); }); } }; export default { hasHeaders, insertToc, updateToc };