/** * Simple WYSIWYG editor for Bootstrap3 * * @category jQuery Plugin * @version 1.1.4 * @author Alexsander Vyshnyvetskyy * @link http://wdmg.github.io/bootstrap-wysiwyg * @copyright Copyright (c) 2019 - 2020 W.D.M.Group, Ukraine * @license https://opensource.org/licenses/MIT Massachusetts Institute of Technology (MIT) License * */ +function($) { "use strict"; var _createClass = (function() { function defineProperties(target, props) { for (var key in props) { var prop = props[key]; prop.configurable = true; if (prop.value) prop.writable = true; } Object.defineProperties(target, props); }; return function(Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; })(); var _classCallCheck = function(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }; var WYSIWYG = (function($) { var className = "wysiwyg"; var _jQueryNoConflict = $.fn[className]; var defaults = { toolbar: [ ['mode'], ['operations', ['undo', 'rendo', 'cut', 'copy', 'paste']], ['styles'], ['fonts', ['select', 'size']], ['text', ['bold', 'italic', 'underline', 'strike', 'subscript', 'superscript', 'font-color', 'bg-color']], ['align', ['left', 'center', 'right', 'justify']], ['lists', ['unordered', 'ordered', 'indent', 'outdent']], ['components', ['table', /*'chart'*/]], ['intervals', ['line-height', 'letter-spacing']], ['insert', ['emoji', 'link', 'image', 'video', 'symbol', /*'bookmark'*/]], ['special', ['print', 'unformat', 'visual', 'clean']], /*['fullscreen'],*/ ], fontSizes: ['8px', '9px', '10px', '11px', '12px', '14px', '15px', '16px', '18px', '20px', '24px', '30px', '32px', '36px', '48px'], fontSizeDefault: '12px', fontFamilies: ['Open Sans', 'Arial', 'Arial Black', 'Courier', 'Courier New', 'Comic Sans MS', 'Helvetica', 'Impact', 'Lucida Grande', 'Lucida Sans', 'Tahoma', 'Times', 'Times New Roman', 'Verdana'], fontFamilyDefault: 'Open Sans', emojiDefault: ["\u{1f600}", "\u{1f62c}", "\u{1f601}", "\u{1f602}", "\u{1f603}", "\u{1f604}", "\u{1f605}", "\u{1f606}", "\u{1f607}", "\u{1f609}", "\u{1f60a}", "\u{1f642}", "\u{1f643}", "\u{1f60b}", "\u{1f60c}", "\u{1f60d}", "\u{1f618}", "\u{1f617}", "\u{1f619}", "\u{1f61a}", "\u{1f61c}", "\u{1f61d}", "\u{1f61b}", "\u{1f911}", "\u{1f913}", "\u{1f60e}", "\u{1f917}", "\u{1f60f}", "\u{1f636}", "\u{1f610}", "\u{1f611}", "\u{1f612}", "\u{1f644}", "\u{1f914}", "\u{1f633}", "\u{1f61e}", "\u{1f61f}", "\u{1f620}", "\u{1f621}", "\u{1f614}", "\u{1f615}", "\u{1f641}", "\u{1f623}", "\u{1f616}", "\u{1f62b}", "\u{1f629}", "\u{1f624}", "\u{1f62e}", "\u{1f631}", "\u{1f628}", "\u{1f630}", "\u{1f62f}", "\u{1f626}", "\u{1f627}", "\u{1f622}", "\u{1f625}", "\u{1f62a}", "\u{1f613}", "\u{1f62d}", "\u{1f635}", "\u{1f632}", "\u{1f910}", "\u{1f637}", "\u{1f912}", "\u{1f915}", "\u{1f634}", "\u{1f4a4}"], symbolsDefault: ["<", ">", "«", "»", "‹", "›", """, "′", "″", "‘", "’", "‚", "“", "”", "„", "❜", "❛", "&", "'", "§", "©", "¬", "®", "¯", "°", "±", "¹", "²", "³", "¼", "½", "¾", "´", "µ", "¶", "·", "¿", "ƒ", "™", "•", "…", "‾", "–", "—", "‰", "}", "{", "=", "≠", "≅", "≈", "≤", "≥", "∠", "⊥", "√", "∑", "∫", "※", "÷", "∞", "@", "[", "]", "←", "↑", "→", "↓", "↔", "↵", "⇐", "⇑", "⇒", "⇓", "⇔", "➠", "➤", "➥", "➦", "➳", "↺", "↻", "⇧", "↩", "⬇", "⬆", "♠", "♣", "♥", "♦", "♡", "♢", "♤", "♧", "₴", "€", "$", "¢", "£", "₽", "¥", "₹", "圓", "₸"], colorPalette: [["rgb(0, 0, 0)","rgb(67, 67, 67)","rgb(102, 102, 102)","rgb(153, 153, 153)","rgb(183, 183, 183)","rgb(204, 204, 204)","rgb(217, 217, 217)","rgb(239, 239, 239)","rgb(243, 243, 243)","rgb(255, 255, 255)"],["rgb(152, 0, 0)","rgb(255, 0, 0)","rgb(255, 153, 0)","rgb(255, 255, 0)","rgb(0, 255, 0)","rgb(0, 255, 255)","rgb(74, 134, 232)","rgb(0, 0, 255)","rgb(153, 0, 255)","rgb(255, 0, 255)"],["rgb(230, 184, 175)","rgb(244, 204, 204)","rgb(252, 229, 205)","rgb(255, 242, 204)","rgb(217, 234, 211)","rgb(208, 224, 227)","rgb(201, 218, 248)","rgb(207, 226, 243)","rgb(217, 210, 233)","rgb(234, 209, 220)","rgb(221, 126, 107)","rgb(234, 153, 153)","rgb(249, 203, 156)","rgb(255, 229, 153)","rgb(182, 215, 168)","rgb(162, 196, 201)","rgb(164, 194, 244)","rgb(159, 197, 232)","rgb(180, 167, 214)","rgb(213, 166, 189)","rgb(204, 65, 37)","rgb(224, 102, 102)","rgb(246, 178, 107)","rgb(255, 217, 102)","rgb(147, 196, 125)","rgb(118, 165, 175)","rgb(109, 158, 235)","rgb(111, 168, 220)","rgb(142, 124, 195)","rgb(194, 123, 160)","rgb(166, 28, 0)","rgb(204, 0, 0)","rgb(230, 145, 56)","rgb(241, 194, 50)","rgb(106, 168, 79)","rgb(69, 129, 142)","rgb(60, 120, 216)","rgb(61, 133, 198)","rgb(103, 78, 167)","rgb(166, 77, 121)","rgb(133, 32, 12)","rgb(153, 0, 0)","rgb(180, 95, 6)","rgb(191, 144, 0)","rgb(56, 118, 29)","rgb(19, 79, 92)","rgb(17, 85, 204)","rgb(11, 83, 148)","rgb(53, 28, 117)","rgb(116, 27, 71)","rgb(91, 15, 0)","rgb(102, 0, 0)","rgb(120, 63, 4)","rgb(127, 96, 0)","rgb(39, 78, 19)","rgb(12, 52, 61)","rgb(28, 69, 135)","rgb(7, 55, 99)","rgb(32, 18, 77)","rgb(76, 17, 48)"]], mode: 'editor', language: 'en-us', translations: {}, highlight: true, debug: false }; const Styles = { 'Header H1': { 'action': 'formatblock', 'value': 'h1', 'wrap': '

', }, 'Header H2': { 'action': 'formatblock', 'value': 'h2', 'wrap': '

', }, 'Header H3': { 'action': 'formatblock', 'value': 'h3', 'wrap': '

', }, 'Header H4': { 'action': 'formatblock', 'value': 'h4', 'wrap': '

', }, 'Header H5': { 'action': 'formatblock', 'value': 'h5', 'wrap': '

', }, 'Header H6': { 'action': 'formatblock', 'value': 'h6', 'wrap': '
', }, 'Paragraph': { 'action': 'formatblock', 'value': 'p', 'wrap': '

', }, 'Blockquote': { 'action': 'formatblock', 'value': 'blockquote', 'wrap': '

', }, 'Preformatted': { 'action': 'formatblock', 'value': 'pre', 'wrap': '
',
            },
            'Div block': {
                'action': 'formatblock',
                'value': 'div',
                'wrap': '
', } } const videoServices = { youtube: 'YouTube', vimeo: 'Vimeo', dailymotion: 'Dailymotion', /*hulu: 'Hulu', twitch: 'Twitch', facebook: 'Facebook', vkontakte: 'vKontakte', twitter: 'Twitter', ustream: 'Ustream',*/ source: 'Source media', /*embed: 'Embed code'*/ }; const urlSchemes = { https: 'https://', http: 'http://', mailto: 'mailto://', ftp: 'ftp://', feed: 'feed://', news: 'news://', tel: 'tel:', skype: 'skype:', telegram: 'tg://', whatsapp: 'whatsapp:', viber: 'viber:', other: 'other' }; const urlLinkTarget = { blank: 'New tab', top: 'Main tab', self: 'Current tab', parent: 'Parent tab', /*iframe: 'Iframe', popup: 'PopUp',*/ }; const urlLinkRel = { nofollow: 'Do not follow (for robots)', noreferrer: 'Do not pass HTTP-referrer', /*archives: 'Link to the site archive', author: 'Link to the page about the author on the same domain', bookmark: 'Permalink to a section or post', first: 'Link to the first page', help: 'Help document link', index: 'Content Link', last: 'Link to the last page', license: 'Link to page with license agreement or copyright', me: 'Link to the author’s page on another domain', next: 'Link to the next page or section', prefetch: 'Indicates that the specified resource must be cached in advance.', prev: 'Link to previous page or section', search: 'Search Link', sidebar: 'Add link to browser favorites', tag: 'Indicates that the label (tag) is related to the current document.', up: 'Link to the parent page', answer: 'Answer to the question', chapter: 'Section or chapter of the current document', co-worker: 'Link to colleague’s page', colleague: 'Link to colleague’s page (not for work)', contact: 'Link to the page with contact information', details: 'Link to the page with details', edit: 'Editable version of the current document', friend: 'Link to friend’s page', question: 'Link to the question page',*/ }; var Editor = (function() { function Editor($element, config) { var _this = this; _classCallCheck(_this, Editor); // Merge default and custom options _this._config = $.extend({}, defaults, config); if (_this._config.debug) console.log('Init WYSIWYG editor...'); // Configure variables _this._editorId = 'wysiwyg-' + (String.fromCharCode(Math.floor(Math.random() * 11)) + Math.floor(Math.random() * 1000000)).trim(); _this._$element = $element instanceof jQuery ? $element : $($element); _this._inputId = _this._$element.attr('id'); // Wrap text input to container _this._$editor = $('
'); _this._$element.wrap(_this._$editor); // Add content to editor _this._$content = $('
'); _this._$content.html(_this._$element.val()); _this._$element.before(_this._$content); _this._source = _this._$element.val(); _this._selection = document.getSelection(); _this._popoverIsVisible = false; _this._$lastFocus = null; // Add toolbar to editor _this._$toolbar = $('
'); _this._$content.before(_this._$toolbar); // Add statusbar to editor _this._$statusbar = $('
'); _this._$statusbar.stat = $(''); _this._$statusbar.path = $(''); _this._$statusbar.append(_this._$statusbar.stat); _this._$statusbar.append(_this._$statusbar.path); _this._$content.after(_this._$statusbar); // Hide input editor _this._$element.addClass('hide'); // Build toolbar by config if(typeof (_this._config.toolbar) == 'object') { $.each(_this._config.toolbar, function (index, elem) { var $toolbar = $('
'); if(elem[0] === 'mode') { // Editor mode switcher var editorButton = _this._buildTollbarButton('mode', 'editor', "fa fa-eye", null, "Editor"); var sourceButton = _this._buildTollbarButton('mode', 'source', "fa fa-code", null, "Source"); if(_this._config.mode == 'editor') editorButton.addClass('active'); else sourceButton.addClass('active'); $toolbar.append(editorButton); $toolbar.append(sourceButton); } else if(elem[0] === 'operations') { // Operations editor controls $toolbar.append(_this._buildTollbarButton('operations', 'undo', "fa fa-reply", null, "Undo")); $toolbar.append(_this._buildTollbarButton('operations', 'rendo', "fa fa-share", null, "Rendo")); $toolbar.append(_this._buildTollbarButton('operations', 'cut', "fa fa-cut", null, "Cut")); $toolbar.append(_this._buildTollbarButton('operations', 'copy', "fa fa-copy", null, "Copy")); $toolbar.append(_this._buildTollbarButton('operations', 'paste', "fa fa-clipboard", null, "Paste")); } else if(elem[0] === 'styles') { // Editor mode switcher $toolbar.append(_this._buildTollbarDropdown('select-style', Styles, "Paragraph", "Text style")); } else if(elem[0] === 'fonts') { // Font select and size if(elem[1].indexOf('select', 0) !== -1) { var fonts = {}; $.each(_this._config.fontFamilies, function(index, value) { fonts[value] = { 'action': 'fontname', 'value': value, 'style': "font-family: " + value + ";" }; }); $toolbar.append(_this._buildTollbarDropdown('font-select', fonts, _this._config.fontFamilyDefault, "Font family")); } if(elem[1].indexOf('size', 0) !== -1) { var sizes = {}; $.each(_this._config.fontSizes, function(index, value) { sizes[value] = { 'action': 'fontsize', 'value': value }; }); $toolbar.append(_this._buildTollbarDropdown('font-size', sizes, _this._config.fontSizeDefault, "Font size")); } } else if(elem[0] === 'text') { // Text decoration if(elem[1].indexOf('bold', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'bold', "fa fa-bold", null, "Bold")); if(elem[1].indexOf('italic', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'italic', "fa fa-italic", null, "Italic")); if(elem[1].indexOf('underline', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'underline', "fa fa-underline", null, "Underline")); if(elem[1].indexOf('strike', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'strike', "fa fa-strikethrough", null, "Striked text")); if(elem[1].indexOf('subscript', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'subscript', "fa fa-subscript", null, "Subscript")); if(elem[1].indexOf('superscript', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'superscript', "fa fa-superscript", null, "Superscript")); if(elem[1].indexOf('bg-color', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'font-color', "fa fa-font", null, "Font color", _this._buildColorPalette(_this._config.colorPalette, "font-color", null))); if(elem[1].indexOf('bg-color', 0) !== -1) $toolbar.append(_this._buildTollbarButton('text', 'bg-color', "fa fa-paint-brush", null, "Background color", _this._buildColorPalette(_this._config.colorPalette, "bg-color", true))); } else if(elem[0] === 'align') { // Text aligment if(elem[1].indexOf('left', 0) !== -1) $toolbar.append(_this._buildTollbarButton('align', 'left', "fa fa-align-left", null, "Align left", null)); if(elem[1].indexOf('center', 0) !== -1) $toolbar.append(_this._buildTollbarButton('align', 'center', "fa fa-align-center", null, "Align center", null)); if(elem[1].indexOf('right', 0) !== -1) $toolbar.append(_this._buildTollbarButton('align', 'right', "fa fa-align-right", null, "Align right", null)); if(elem[1].indexOf('justify', 0) !== -1) $toolbar.append(_this._buildTollbarButton('align', 'justify', "fa fa-align-justify", null, "Justify content", null)); } else if(elem[0] === 'lists') { // Lists && outdent if(elem[1].indexOf('unordered', 0) !== -1) $toolbar.append(_this._buildTollbarButton('lists', 'unordered', "fa fa-list-ul", null, "Unordered list")); if(elem[1].indexOf('ordered', 0) !== -1) $toolbar.append(_this._buildTollbarButton('lists', 'ordered', "fa fa-list-ol", null, "Ordered list")); if(elem[1].indexOf('indent', 0) !== -1) $toolbar.append(_this._buildTollbarButton('lists', 'indent', "fa fa-indent", null, "Indent")); if(elem[1].indexOf('outdent', 0) !== -1) $toolbar.append(_this._buildTollbarButton('lists', 'outdent', "fa fa-outdent", null, "Outdent")); } else if(elem[0] === 'components') { // Components if(elem[1].indexOf('table', 0) !== -1) $toolbar.append(_this._buildTollbarButton('components', 'table', "fa fa-table", null, "Insert table", _this._buildTableGrid())); if(elem[1].indexOf('chart', 0) !== -1) $toolbar.append(_this._buildTollbarButton('components', 'chart', "fa fa-pie-chart", null, "Add chart")); } else if(elem[0] === 'intervals') { // Text properties if(elem[1].indexOf('line-height', 0) !== -1) $toolbar.append(_this._buildTollbarButton('interval', 'line-height', "fa fa-text-height", null, "Lines interval", _this._buildLineHeightList())); if(elem[1].indexOf('letter-spacing', 0) !== -1) $toolbar.append(_this._buildTollbarButton('interval', 'letter-spacing', "fa fa-text-width", null, "Letter spacing", _this._buildLetterSpacingList())); } else if(elem[0] === 'insert') { // Inserts if(elem[1].indexOf('emoji', 0) !== -1) $toolbar.append(_this._buildTollbarButton('insert', 'emoji', "fa fa-smile fa-smile-o", null, "Add emoji", _this._buildEmojiList())); if(elem[1].indexOf('link', 0) !== -1) $toolbar.append(_this._buildTollbarButton('insert', 'link', "fa fa-link", null, "Add URL", _this._buildUrlForm('link'))); if(elem[1].indexOf('image', 0) !== -1) $toolbar.append(_this._buildTollbarButton('insert', 'image', "fa fa-image", null, "Add image", _this._buildUrlForm('image'))); if(elem[1].indexOf('video', 0) !== -1) $toolbar.append(_this._buildTollbarButton('insert', 'video', "fa fa-video-camera fa-video", null, "Add video", _this._buildUrlForm('video'))); if(elem[1].indexOf('symbol', 0) !== -1) $toolbar.append(_this._buildTollbarButton('insert', 'symbol', "fa fa-hashtag", null, "Add symbol", _this._buildSymbolsList())); if(elem[1].indexOf('bookmark', 0) !== -1) $toolbar.append(_this._buildTollbarButton('insert', 'bookmark', "fa fa-bookmark", null, "Add bookmark")); } else if(elem[0] === 'special') { // Inserts if(elem[1].indexOf('print', 0) !== -1) $toolbar.append(_this._buildTollbarButton('special', 'print', "fa fa-print", null, "Print")); if(elem[1].indexOf('clean', 0) !== -1) $toolbar.append(_this._buildTollbarButton('special', 'clean', "fa fa-eraser", null, "Erase style")); if(elem[1].indexOf('visual', 0) !== -1) $toolbar.append(_this._buildTollbarButton('special', 'visual', "fa fa-solar-panel", null, "Visual blocks")); if(elem[1].indexOf('unformat', 0) !== -1) $toolbar.append(_this._buildTollbarButton('special', 'unformat', "fa fa-trash-o fa-trash-alt", null, "Clear HTML")); } else if(elem[0] === 'fullscreen') { // Fullscreen mode $toolbar.addClass('pull-right'); $toolbar.append(_this._buildTollbarButton('fullscreen', true, "fa fa-arrows-alt", null, "Fullscreen mode")); } _this._$toolbar.append($toolbar); }); } // Set behavior for toolbar buttons if(_this._$toolbar.length) { _this._$toolbar.on('click', '[data-action]', function(event) { var $target = $(event.currentTarget); var action = $target.data('action'); var selection = _this._selection; var value = $target.data('value'); var data = $target.data(); if (typeof (action) !== 'undefined' && typeof (value) !== 'undefined') { if (_this._config.debug) console.log('Switch action: `' + action + '` with value: `' + value + '`'); switch (action) { case 'mode': switch (value) { case 'editor': if (_this._config.mode !== value) { _this._config.mode = value; _this._$content.html(_this._source); _this._$toolbar.find('[data-action="mode"]').removeClass('active'); _this._$toolbar.find('[data-action="mode"][data-value="editor"]').addClass('active'); _this._$content.addClass('editor-mode').removeClass('source-mode'); _this._$content.focus(); } _this._$toolbar.find('.btn-group').removeClass('hide'); break; case 'source': _this._$toolbar.find('.btn-group').not('#toolbarGroup-' + action).addClass('hide'); if (_this._config.mode !== value) { _this._config.mode = value; _this._source = _this._$content.html(); var $source = $('
');
                                                $source.text(_this._source);

                                                _this._$content.html(_this._trimSource($source.html()));

                                                if (_this._config.highlight) {
                                                    hljs.initHighlighting.called = false;
                                                    hljs.configure({
                                                        useBR: true,
                                                        languages: ['html', 'javascript', 'css']
                                                    });
                                                    hljs.highlightBlock(_this._$content.get(0));
                                                }

                                                _this._$toolbar.find('[data-action="mode"]').removeClass('active');
                                                _this._$toolbar.find('[data-action="mode"][data-value="source"]').addClass('active');

                                                _this._$content.removeClass('editor-mode').addClass('source-mode');
                                                _this._$content.focus();
                                            }
                                            break;
                                    }
                                    break;

                                case 'formatblock':
                                    _this._formatDoc('formatblock', value);
                                    break;

                                case 'fontname':
                                    _this._formatDoc('fontname', value);
                                    break;

                                case 'fontsize':
                                    _this._selection.anchorNode.parentElement.removeAttribute("size");
                                    _this._selection.anchorNode.parentElement.style.fontSize = value;
                                    break;

                                case 'style':
                                    var styles = _this._selection.anchorNode.parentElement.style.cssText;

                                    if(styles)
                                        styles += value;
                                    else
                                        styles = value;

                                    _this._selection.anchorNode.parentElement.removeAttribute("style");
                                    _this._selection.anchorNode.parentElement.style = styles;
                                    break;

                                case 'fullscreen':

                                    if (_this._config.debug)
                                        console.log('Fire action: ' + action + ' with value: ' + value + ' is not supported.');

                                    break;

                                case 'operations':
                                    switch (value) {
                                        case 'undo':
                                            _this._formatDoc('undo');
                                            break;

                                        case 'rendo':
                                            _this._formatDoc('rendo');
                                            break;
                                        case 'cut':
                                            _this._formatDoc('cut');
                                            break;

                                        case 'copy':
                                            _this._formatDoc('copy');
                                            break;

                                        case 'paste':
                                            _this._formatDoc('paste');
                                            break;
                                    }
                                    break;

                                case 'text':
                                    switch (value) {
                                        case 'bold':
                                            _this._formatDoc('bold');
                                            break;

                                        case 'italic':
                                            _this._formatDoc('italic');
                                            break;

                                        case 'underline':
                                            _this._formatDoc('underline');
                                            break;

                                        case 'strike':
                                            _this._formatDoc('strikeThrough');
                                            break;

                                        case 'subscript':
                                            _this._formatDoc('subscript');
                                            break;

                                        case 'superscript':
                                            _this._formatDoc('superscript');
                                            break;

                                    }
                                    break;

                                case 'font-color':
                                    if(value == 'unset') {

                                        if(_this._selection.anchorNode)
                                            _this._selection.anchorNode.parentElement.style.backgroundColor = "";

                                        if(_this._selection.anchorNode.parentElement.style.length)
                                            _this._selection.anchorNode.parentElement.removeAttribute("style");

                                    } else {
                                        _this._$toolbar.find('[data-action="text"][data-value="font-color"] > span').css('border-bottom-color', value);
                                        _this._formatDoc('foreColor', value);
                                    }
                                    break;

                                case 'bg-color':
                                    if(value == 'unset') {

                                        if(_this._selection.anchorNode)
                                            _this._selection.anchorNode.parentElement.style.backgroundColor = "";

                                        if(_this._selection.anchorNode.parentElement.style.length)
                                            _this._selection.anchorNode.parentElement.removeAttribute("style");

                                    } else {
                                        _this._$toolbar.find('[data-action="text"][data-value="bg-color"] > span').css('border-bottom-color', value);
                                        _this._formatDoc('hiliteColor', value);
                                    }
                                    break;

                                case 'align':
                                    switch (value) {
                                        case 'left':
                                            _this._formatDoc('justifyLeft');
                                            break;

                                        case 'center':
                                            _this._formatDoc('justifyCenter');
                                            break;

                                        case 'right':
                                            _this._formatDoc('justifyRight');
                                            break;

                                        case 'justify':
                                            _this._formatDoc('justifyFull');
                                            break;
                                    }
                                    break;


                                case 'lists':
                                    switch (value) {
                                        case 'unordered':
                                            _this._formatDoc('insertUnorderedList');
                                            break;

                                        case 'ordered':
                                            _this._formatDoc('insertOrderedList');
                                            break;

                                        case 'indent':
                                            _this._formatDoc('indent');
                                            break;

                                        case 'outdent':
                                            _this._formatDoc('outdent');
                                            break;
                                    }
                                    break;

                                case 'insert-table':
                                    if(_this._selection.anchorNode) {
                                        var options = value.split('|', 2);
                                        var $parent = $(_this._selection.anchorNode.parentElement);
                                        var content = _this._generateTable(parseFloat(options[0]), parseFloat(options[2]));
                                        $parent.after(content);
                                    }
                                    break;

                                case 'components':
                                    switch (value) {

                                        case 'chart':
                                            if (_this._config.debug)
                                                console.log('Fire action: ' + action + ' with value: ' + value + ' is not supported.');

                                            break;
                                    }
                                    break;

                                case 'line-height':
                                    if(_this._selection.anchorNode) {
                                        var lineHeight = parseFloat(value) * 100 + "%";

                                        if (parseFloat(value) == 0)
                                            _this._selection.anchorNode.parentElement.style.lineHeight = "inherit";
                                        else
                                            _this._selection.anchorNode.parentElement.style.lineHeight = lineHeight;
                                    }
                                    break;

                                case 'letter-spacing':
                                    if(_this._selection.anchorNode) {
                                        var letterSpacing = parseFloat(value) + "px";

                                        if (parseFloat(value) == 0)
                                            _this._selection.anchorNode.parentElement.style.letterSpacing = "inherit";
                                        else
                                            _this._selection.anchorNode.parentElement.style.letterSpacing = letterSpacing;

                                    }
                                    break;

                                case 'add-url':
                                    var text = _this._selection.toString();
                                    if(_this._selection && text) {

                                        var url = value;
                                        console.log(data);

                                        if (data.scheme) {
                                            if (data.scheme == 'https')
                                                url = 'https://' + url;
                                            else if (data.scheme == 'http')
                                                url = 'http://' + url;
                                            else if (data.scheme == 'mailto')
                                                url = 'mailto://' + url;
                                            else if (data.scheme == 'ftp')
                                                url = 'ftp://' + url;
                                            else if (data.scheme == 'feed')
                                                url = 'feed://' + url;
                                            else if (data.scheme == 'news')
                                                url = 'news://' + url;
                                            else if (data.scheme == 'tel')
                                                url = 'tel:' + url;
                                            else if (data.scheme == 'skype')
                                                url = 'skype:' + url;
                                            else if (data.scheme == 'tg')
                                                url = 'tg://' + url;
                                            else if (data.scheme == 'whatsapp')
                                                url = 'whatsapp:' + url;
                                            else if (data.scheme == 'viber')
                                                url = 'viber:' + url;
                                        }

                                        var title = '';
                                        if (data.title) {
                                            title = ' title="' + data.title + '"';
                                        }

                                        var className = '';
                                        if (data.class) {
                                            className = ' class="' + data.class + '"';
                                        }

                                        var target = '';
                                        if (data.target) {
                                            if (data.target == 'blank')
                                                target = ' target="_blank"';
                                            else if (data.target == 'top')
                                                target = ' target="_top"';
                                            else if (data.target == 'self')
                                                target = ' target="_self"';
                                            else if (data.target == 'parent')
                                                target = ' target="_parent"';
                                        }

                                        var rel = '';
                                        if (data.relation) {
                                            if (data.relation == 'nofollow')
                                                rel = ' rel="nofollow"';
                                            else if (data.relation == 'noreferrer')
                                                rel = ' rel="noreferrer"';
                                        }

                                        var $link = $('');
                                        $link.text(text);

                                        var range = _this._selection.getRangeAt(0);
                                        range.deleteContents();
                                        range.insertNode($link.get(0));
                                    }
                                    break;

                                case 'add-video':
                                    if(_this._selection && value) {

                                        var url = value;
                                        if (data.service) {
                                            if (data.service == 'youtube') {
                                                var videoId = null;
                                                var regExp = /^.*(youtube\/|youtu.be\/|v\/|u\/\w\/|embed\/|watch\?v=|\&v=)([^#\&\?]*).*/;
                                                var match = url.match(regExp);

                                                if (match !== null && match[2].length == 11)
                                                    videoId = match[2];

                                                if (videoId)
                                                    url = 'https://www.youtube.com/embed/' + videoId;

                                            } else if (data.service == 'vimeo') {
                                                var videoId = null;
                                                var regExp = /(?:www\.|player\.)?vimeo.com\/(?:channels\/(?:\w+\/)?|groups\/(?:[^\/]*)\/videos\/|album\/(?:\d+)\/video\/|video\/|)(\d+)(?:[a-zA-Z0-9_\-]+)?/i;
                                                var match = url.match(regExp);

                                                if (match !== null && match[1].length == 9)
                                                    videoId = match[1];

                                                if (videoId)
                                                    url = 'https://player.vimeo.com/video/' + match[1];

                                            } else if (data.service == 'dailymotion') {
                                                var videoId = null;
                                                var regExp = /^.+dailymotion.com\/(video|hub)\/([^_]+)[^#]*(#video=([^_&]+))?/;
                                                var match = url.match(regExp);

                                                if (match !== null) {
                                                    if (match[4] !== undefined) {
                                                        videoId = match[4];
                                                    }
                                                    videoId = match[2];
                                                }

                                                if (videoId)
                                                    url = 'https://www.dailymotion.com/embed/video/' + videoId;

                                            }
                                        }

                                        var $embed = $('
' + '' + '
'); var range = _this._selection.getRangeAt(0); range.deleteContents(); range.insertNode($embed.get(0)); } break; case 'add-image': if(_this._selection && value) { var $image = $(''); var range = _this._selection.getRangeAt(0); range.deleteContents(); range.insertNode($image.get(0)); } break; case 'insert-html': _this._formatDoc('insertHTML', value); break; case 'special': switch (value) { case 'print': _this._printDoc(); break; case 'clean': _this._formatDoc('removeFormat'); break; case 'visual': if (_this._$content.hasClass('visual')) { _this._$toolbar.find('[data-action="special"][data-value="visual"]').removeClass('active'); _this._$content.removeClass('visual'); _this._$content.focus(); } else { _this._$toolbar.find('[data-action="special"][data-value="visual"]').addClass('active'); _this._$content.addClass('visual'); _this._$content.focus(); } break; case 'unformat': _this._formatDoc('selectAll'); _this._formatDoc('removeFormat'); var string = _this._$content.html(); string = _this._stripTags(string); string = string.replace(/(\r\n|\n|\r)/g, ''); string = string.replace(//g, '
'); _this._$content.html(string); break; } break; default: if (_this._config.debug) console.warn('Unrecognized action: ' + action + ' with value: ' + value); break; } } }); } // On selected content _this._$content.on('mouseup click focus', function (event) { const $this = $(this); if (_this._popoverIsVisible) _this._hideAllPopovers(); if (event.target.type !== "text") _this._selection = document.getSelection(); if (_this._selection.getRangeAt && _this._selection.rangeCount) { if (_this._selection.parentNode) { var $target = $(_this._selection.parentNode); _this._updateState($target); } else if (_this._selection.parentElement) { var $target = $(_this._selection.parentElement); _this._updateState($target); } } if (_this._config.debug) console.log('Current selection: ', _this._selection); var $target = $(event.target); _this._updateState($target); $this.trigger('change'); }); // On click or keydown from content area _this._$content.on('keydown', function (event) { const $this = $(this); if(_this._popoverIsVisible) _this._hideAllPopovers(); if(event.target.type !== "text") _this._selection = document.getSelection(); if (_this._selection.getRangeAt && _this._selection.rangeCount) { if (_this._selection.parentNode) { var $target = $(_this._selection.parentNode); _this._updateState($target); } else if (_this._selection.parentElement) { var $target = $(_this._selection.parentElement); _this._updateState($target); } } var $target = $(event.target); _this._updateState($target); $this.trigger('change'); if (_this._config.debug) console.log('Keydown fired: ' + event.keyCode); }); // On content change _this._$content.on('change', function(event) { const $this = $(this); setTimeout(function() { if (_this._config.mode == 'editor') _this._source = $this.html(); else _this._source = $this.text(); if ($(_this._$element).is("textarea")) _this._$element.html(_this._source); else _this._$element.val(_this._source); if (_this._config.debug) console.log('Content change...'); }, 200); }); // On content lost focus _this._$content.on('blur', function() { const $this = $(this); _this._$lastFocus = this; _this._selection = document.getSelection(); $this.trigger('change'); if (_this._config.debug) console.log('Content lost focus: ', _this._selection); }); // Set focus on content _this._$content.focus(); } _createClass(Editor, { element: { value: function element() { var _this = this; return _this._$element; } }, _replaceAll: { value: function replaceAll(search, replace, string) { return string.split(search).join(replace); } }, _stripTags: { value: function stripTags(string, tags) { var key, allowed_tags = []; if (tags) allowed_tags = tags.match(/([a-zA-Z]+)/gi); if (typeof (string) !== 'string') string = string.toString(); var matches = string.match(/(<\/?[\S][^>]*>)/gi); for (key in matches) { if (isNaN(key)) continue; var html = matches[key].toString(); var allowed = false; for (key in allowed_tags) { var tag = allowed_tags[key]; var i = html.toLowerCase().indexOf('<'+ tag +'>'); if (i != 0) i = html.toLowerCase().indexOf('<'+ tag +' '); if (i != 0) i = html.toLowerCase().indexOf(' 1) { var allSiblings = parent.children(); var index = allSiblings.index(realNode) + 1; if (index > 1) { name += ':nth-child(' + index + ')'; } } } tags.push(name); var id = $(realNode).attr("id"); if (id) name += "#" + id; var classname = $(realNode).attr("class"); if (classname) name += "." + classname.replace(/\./g, '.'); path = name + (path ? ' > ' + path : ''); node = parent; } return { path: path, tags: tags }; } }, _getTextStat: { value: function getTextStat(el) { var words = 0, length = 0, chars = 0, normalizedValue; var isContentEditable = el && el.contentEditable; if (isContentEditable) normalizedValue = el.innerText.replace(/\r\n/g, "\n"); else normalizedValue = el.value.replace(/\r\n/g, "\n"); words = this._stripTags(normalizedValue).split(' ').length; length = normalizedValue.length; chars = this._trimSource(normalizedValue.replace(/\s/g, "")).length; return { words: words, length: length, chars: chars } } }, _getTextPosition: { value: function getCursorPosition(el) { var line = 0, start = 0, end = 0, selected = 0, normalizedValue, range, textInputRange, len, endRange; var isContentEditable = el && el.contentEditable; if ("selectionStart" in el && document.activeElement == el) { start = el.selectionStart; end = el.selectionEnd; normalizedValue = el.value.replace(/\r\n/g, "\n"); line = normalizedValue.substr(0, el.selectionStart).split("\n").length; } else if (isContentEditable) { start = window.getSelection().getRangeAt(0).startOffset; end = window.getSelection().getRangeAt(0).endOffset; normalizedValue = el.innerText.replace(/\r\n/g, "\n"); line = (normalizedValue.substr(0, el.selectionStart).split("\n").length - 1); if(line == 0) line = 1; } else { range = this._selection.createRange(); if (range && range.parentElement() == el) { len = el.value.length; normalizedValue = el.value.replace(/\r\n/g, "\n"); // Create a working TextRange that lives only in the input textInputRange = el.createTextRange(); textInputRange.moveToBookmark(range.getBookmark()); // Check if the start and end of the selection are at the very end // of the input, since moveStart/moveEnd doesn't return what we want // in those cases endRange = el.createTextRange(); endRange.collapse(false); if (textInputRange.compareEndPoints("StartToEnd", endRange) > -1) { start = end = len; } else { start = -textInputRange.moveStart("character", -len); start += normalizedValue.slice(0, start).split("\n").length - 1; if (textInputRange.compareEndPoints("EndToEnd", endRange) > -1) { end = len; } else { end = -textInputRange.moveEnd("character", -len); end += normalizedValue.slice(0, end).split("\n").length - 1; } } } } selected = (this._selection.toString()).length; return { line: line, start: start, end: end, selected: selected } } }, _formatDoc: { value: function formatDoc(command, value) { document.execCommand(command, false, value); this._$content.focus(); } }, _printDoc: { value: function printDoc() { var print = window.open("","_blank","width=450,height=470,left=400,top=100,menubar=yes,toolbar=no,location=no,scrollbars=yes"); print.document.open(); print.document.write("Print<\/title><\/head><body onload=\"print();\">" + this._$content.get(0).innerHTML + "<\/body><\/html>"); print.document.close(); } }, _hideAllPopovers: { value: function hideAllPopovers() { this._$toolbar.find('.popover').each(function() { $(this).popover('hide'); }); this._popoverIsVisible = false; } }, _detectLanguage: { value: function detectLanguage() { var language = null; if (navigator.languages && navigator.languages.length) { language = navigator.languages[0]; } else { language = navigator.userLanguage || navigator.language || navigator.browserLanguage || 'en'; } return language.toLowerCase(); } }, _translate: { value: function translate(string) { var _this = this; var language = _this._config.language; if (typeof (language) === "undefined") language = _this._detectLanguage(); if (_this._config.translations.hasOwnProperty(language)) { if (_this._config.translations[language][string.toString()]) { string = _this._config.translations[language][string.toString()]; } } return string.toLocaleString(); } }, _buildTollbarButton: { value: function buildTollbarButton(action, value, icon, hotkey, tooltip, content) { var _this = this; var selection = _this._selection; var $button = $('<button type="button" class="btn btn-default" tabindex="-1" />'); if (action) $button.attr('data-action', action); if (value) $button.attr('data-value', value); if (hotkey) $button.attr('data-hotkey', hotkey); if (tooltip) { $button.tooltip({ html: true, placement: 'top', container: 'body', title: _this._translate(tooltip.toString().trim()) }); } if (content) { $button.popover({ html: true, trigger: 'manual', viewport: 'body', placement: 'bottom', content: function() { if (typeof (content) === "object") return content; else return $(content); } }).on('shown.bs.popover', function(event) { var popoverId = $(event.target).attr('aria-describedby'); var $popover = _this._$toolbar.find('#'+popoverId); var selection = _this._selection; var range = selection.getRangeAt(0); if(selection && range) { $popover.find('input').on('blur', function(event) { if (event.target.type == "text") { if (_this._$lastFocus) { setTimeout(function() { _this._$lastFocus.focus(); _this._selectText(selection, range); }, 50); } } return false; }); } $popover.on('click', function(event) { event.preventDefault(); if (_this._config.debug) { console.log('Popover event target type: ' + event.target.type); console.log('Popover event target tag: ' + event.target.tagName); } if (!(event.target.type) && !(event.target.tagName.toLowerCase() == 'a')) return; if ($(event.target).get(0).hasAttribute('data-action')) { $popover.popover('hide'); } }); if ($popover.find('.table-grid').length) { $popover.find('.table-grid tr > td').hover(function() { $(this).addClass('selected'); $(this).prevAll().addClass('selected'); $(this).parent().prevAll().find('td:lt('+ ($(this).index() + 1) + ')').addClass('selected'); }, function() { $(this).removeClass('selected'); $(this).prevAll().removeClass('selected'); $(this).parent().prevAll().find('td:lt('+ ($(this).index() + 1) + ')').removeClass('selected'); }); } }).on('click', function(event) { if (_this._config.debug) { console.log('Element event target type: ' + event.target.type); console.log('Element event target tag: ' + event.target.tagName); } event.preventDefault(); event.stopPropagation(); if(_this._popoverIsVisible) _this._hideAllPopovers(); $button.popover('show'); _this._popoverIsVisible = true; }); } else { $button.on('click', function(event) { event.preventDefault(); _this._hideAllPopovers(); }); } if (icon) $button.append('<span class="' + icon + '" />'); return $button; } }, _buildTollbarDropdown: { value: function buildTollbarDropdown(action, list, label, tooltip) { var $dropdown = $('<div class="dropdown" />'); var $dropdownBtn = $('<button type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown" />'); var $dropdownMenu = $('<ul class="dropdown-menu" />'); var $dropdownItem = $('<li />'); var $dropdownLink = $('<a href="#" tabindex="-1" />'); if (typeof (list) == "object") { $.each(list, function(index, elem) { var $link = $dropdownLink.clone(); var $item = $dropdownItem.clone(); if (typeof (elem) == 'object') { if (elem['action']) $link.attr('data-action', elem['action']); if (elem['value']) $link.attr('data-value', elem['value']); if (elem['wrap']) $link.html($(elem['wrap']).text(index)); else $link.text(index); if (elem['style']) $link.attr('style', elem['style'].toString()); if(index == label) $item.addClass('active'); $item.append($link); $dropdownMenu.append($item); } else { $link.text(elem); $link.attr('data-action', action); $link.attr('data-value', elem); if(elem == label) $item.addClass('active'); $item.append($link); $dropdownMenu.append($item); } $link.on('click', function (e) { e.preventDefault(); }); }); } if (label) $dropdownBtn.text(label + ' '); else $dropdownBtn.text('Dropdown '); $dropdownBtn.append('<b class="caret" />'); if (tooltip) { $dropdownBtn.tooltip({ html: true, placement: 'top', title: this._translate(tooltip.toString().trim()) }); } $dropdown.append($dropdownBtn); $dropdown.append($dropdownMenu); return $dropdown; } }, _buildColorPalette: { value: function buildColorPalette(palette, action, reset) { var content = ''; $.each(palette, function (outer, colors) { content += '<table class="color-palette"><tr>'; $.each(colors, function (inner, color) { content += '<td><a href="#" data-action="' + action + '" data-value="' + color + '" style="background-color: ' + color + '"> </a></td>' content += ((parseInt(inner) + 1)%10 ? '' : '</tr>'); content += ((parseInt(inner) + 1)%10 ? '' : '<tr>'); }); content += '</tr></table>'; }); if(reset) content += '<p><a href="#" class="btn btn-sm btn-block" data-action="' + action + '" data-value="unset">Reset color</a></p>'; return content; } }, _buildTableGrid: { value: function buildColorPalette() { var content = '<table class="table-grid">'; for(var row = 1; row <= 6; row++) { content += '<tr>'; for(var column = 1; column <= 8; column++) { content += '<td><a href="#" data-action="insert-table" data-value="' + row + '|'+ column +'"> </a></td>' } content += '</tr>'; } content += '</table>'; return content; } }, _buildLetterSpacingList: { value: function buildLetterSpacingList() { var content = '<ul class="nav nav-pills nav-stacked">\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="-5">-5</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="-3">-3</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="-2">-2</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="-1">-1</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="0">0</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="1">1</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="2">2</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="3">3</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="5">5</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="8">8</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="10">10</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="12">12</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="15">15</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="25">25</a></li>\n' + ' <li role="presentation"><a href="#" data-action="letter-spacing" data-value="50">50</a></li>\n' + '</ul>'; return content; } }, _buildLineHeightList: { value: function buildLineHeightList() { var content = '<ul class="nav nav-pills nav-stacked">\n' + ' <li role="presentation"><a href="#" data-action="line-height" data-value="0.5">0.5</a></li>\n' + ' <li role="presentation"><a href="#" data-action="line-height" data-value="1.0">1.0</a></li>\n' + ' <li role="presentation"><a href="#" data-action="line-height" data-value="1.15">1.15</a></li>\n' + ' <li role="presentation"><a href="#" data-action="line-height" data-value="1.5">1.5</a></li>\n' + ' <li role="presentation"><a href="#" data-action="line-height" data-value="2.0">2.0</a></li>\n' + '</ul>'; return content; } }, _selectText: { value: function selectText(selection, range) { if(!selection) selection = document.getSelection(); if(!range) range = selection.getRangeAt(0); selection.removeAllRanges(); selection.addRange(range); } }, _buildEmojiList: { value: function buildEmojiList() { var emojis = this._config.emojiDefault; if(emojis.length > 0) { var maxRows = Math.round(emojis.length / 8)+1; var content = '<table class="emojis-list">'; for(var row = 1, index = 0; row <= maxRows; row++) { content += '<tr>'; for(var column = 1; column <= 8; column++) { if(index == emojis.length) break; content += '<td><a href="#" data-action="insert-html" data-value="' + emojis[index].toString() + '">'+ emojis[index] +'</a></td>'; index++; } content += '</tr>'; } content += '</table>'; return content; } else { return false; } } }, _buildSymbolsList: { value: function buildSymbolsList() { var symbols = this._config.symbolsDefault; if(symbols.length > 0) { var maxRows = Math.round(symbols.length / 10); var content = '<table class="symbols-list">'; for(var row = 1, index = 0; row <= maxRows; row++) { content += '<tr>'; for(var column = 1; column <= 10; column++) { if(index == symbols.length) break; content += '<td><a href="#" data-action="insert-html" data-value="' + symbols[index] + '" style="min-width:16px;text-align:center;">'+ symbols[index] +'</a></td>'; index++; } content += '</tr>'; } content += '</table>'; return content; } else { return false; } } }, _buildDdropdown: { value: function buildDdropdown(dropdownId, buttonText, buttonCaret, menuItems, defaultValue, dataAttr) { if (buttonText == null) buttonText = 'Not set'; if (buttonCaret == null) buttonCaret = '<span class="caret"></span>'; if (menuItems == null) buttonCaret = {}; if (dataAttr == null) dataAttr = 'data-value'; var $dropdown = $('<div class="dropdown" />'); var $dropdownButton = $('<button type="button" class="btn btn-block btn-default dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" />'); $dropdownButton.attr('id', dropdownId); $dropdownButton.html(buttonText + ' ' + buttonCaret); // Build dropdown menu var $dropdownMenu = $('<ul class="dropdown-menu" aria-labelledby="' + dropdownId + '" />'); if (!defaultValue) $dropdownMenu.append('<li class="active"><a href="#">' + buttonText + '</a></li>'); for (let [id, name] of Object.entries(menuItems)) { if (id.toString() == 'separator') { $dropdownMenu.append('<li role="separator" class="divider"></li>'); } else if (defaultValue == id.toString()) { $dropdownMenu.append('<li class="active"><a href="#" ' + dataAttr + '="' + id + '">' + name + '</a></li>'); } else { $dropdownMenu.append('<li><a href="#" ' + dataAttr + '="' + id + '">' + name + '</a></li>'); } } // Click by dropdown menu items $dropdownMenu.find('li').on('click', function(event) { $dropdownMenu.find('li').removeClass('active'); $(this).addClass('active'); $dropdownButton.html($(this).text() + ' ' + buttonCaret); }); $dropdown.append($dropdownButton); $dropdown.append($dropdownMenu); return $dropdown; } }, _buildUrlForm: { value: function buildUrlForm(type) { var _this = this; if (this._config.debug) console.log('Build URL form for type: ' + type); var $form = $('<form class="form-horizontal" />'); var $formGroup = $('<div class="form-group" />'); var $container = $('<div class="col-xs-12 col-sm-12" />'); var $inputGroup = $('<div class="input-group input-group-sm" />'); var $dropdown = $('<div class="dropdown" />'); var $dropdownButton = $('<button type="button" class="btn btn-block btn-secondary dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" />'); if (type == "image") { $inputGroup.append('<span class="input-group-addon">Image:</span>'); } else if (type == "video") { var $dropdown = _this._buildDdropdown('videoServices', 'YouTube', null, videoServices, 'youtube', 'data-service'); $dropdown.attr('class', 'input-group-btn'); $dropdown.find('.btn[data-toggle="dropdown"]').toggleClass('btn-default', 'btn-secondary'); $inputGroup.append($dropdown); } else if (type == "link") { var $dropdown = _this._buildDdropdown('urlSchemes', 'https://', null, urlSchemes, 'https', 'data-scheme'); $dropdown.attr('class', 'input-group-btn'); $dropdown.find('.btn[data-toggle="dropdown"]').toggleClass('btn-default', 'btn-secondary'); $inputGroup.append($dropdown); } var $input = $('<input type="text" class="form-control" placeholder="Type your URL..." />'); if (type == "image") { $input.attr('id', "imageUrl"); } else if (type == "video") { $input.attr('id', "videoUrl"); } else if (type == "link") { $input.attr('id', "urlInput"); } $inputGroup.append($input); var $buttonWrap = $('<span class="input-group-btn" />'); var $button = $('<button type="button" class="btn btn-block btn-primary">Add</button>'); if (type == "image") { $button.attr('data-action', "add-image"); } else if (type == "video") { $button.attr('data-action', "add-video"); } else { $button.attr('data-action', "add-url"); } $button.on('click', function (event) { var action = $(event.target).data('action'); $(event.target).data('value', $input.val()); if (action == "add-url") { var urlScheme = $form.find('[aria-labelledby="urlSchemes"] li.active a[data-scheme]').first().data('scheme'); $(event.target).data('scheme', (urlScheme) ? urlScheme : null); var urlTile = $form.find('#urlTile').val(); $(event.target).data('title', (urlTile) ? urlTile : null); var urlClass = $form.find('#urlClass').val(); $(event.target).data('class', (urlClass) ? urlClass : null); var urlLinkTarget = $form.find('[aria-labelledby="urlLinkTarget"] li.active a[data-target]').first().data('target'); $(event.target).data('target', (urlLinkTarget) ? urlLinkTarget : null); var urlLinkRel = $form.find('[aria-labelledby="urlLinkRel"] li.active a[data-rel]').first().data('rel'); $(event.target).data('relation', (urlLinkRel) ? urlLinkRel : null); } else if (action == "add-video") { var videoService = $form.find('[aria-labelledby="videoServices"] li.active a[data-service]').first().data('service'); $(event.target).data('service', (videoService) ? videoService : null); } console.log($button.data()); }); $buttonWrap.append($button); $inputGroup.append($buttonWrap); $container.append($inputGroup); $formGroup.append($container); $form.append($formGroup); if (type == "link") { var content = ''; content += '<div class="form-group form-group-sm">'; content += ' <div class="form-row">'; content += ' <label class="col-xs-12 col-sm-4">Tile:</label>'; content += ' <div class="col-xs-12 col-sm-8">'; content += ' <input type="text" class="form-control" id="urlTile" placeholder="Title of link..." />'; content += ' </div>'; content += ' </div>'; content += '</div>'; content += '<div class="form-group form-group-sm">'; content += ' <div class="form-row">'; content += ' <label class="col-xs-12 col-sm-4">Class:</label>'; content += ' <div class="col-xs-12 col-sm-8">'; content += ' <input type="text" class="form-control" id="urlClass" placeholder="CSS class name..." />'; content += ' </div>'; content += ' </div>'; content += '</div>'; $form.append(content); // Link target var $formGroup = $('<div class="form-group form-group-sm" />'); var $inputGroup = $('<div class="form-row input-group-sm" />'); var $inputLabel = $('<label class="col-xs-12 col-sm-4" />'); $inputLabel.text('Target:'); $inputGroup.append($inputLabel); var $dropdown = _this._buildDdropdown('urlLinkTarget', 'Not set', null, urlLinkTarget, null, 'data-target'); $dropdown.addClass('col-xs-12 col-sm-8'); $dropdown.find('.btn[data-toggle="dropdown"]').addClass('btn-sm'); $inputGroup.append($dropdown); $formGroup.append($inputGroup); $form.append($formGroup); // Link rel var $formGroup = $('<div class="form-group form-group-sm" />'); var $inputGroup = $('<div class="form-row input-group-sm" />'); var $inputLabel = $('<label class="col-xs-12 col-sm-4" />'); $inputLabel.text('Relation:'); $inputGroup.append($inputLabel); var $dropdown = _this._buildDdropdown('urlLinkRel', 'Not set', null, urlLinkRel, null, 'data-rel') $dropdown.addClass('col-xs-12 col-sm-8'); $dropdown.find('.btn[data-toggle="dropdown"]').addClass('btn-sm'); $inputGroup.append($dropdown); $formGroup.append($inputGroup); $form.append($formGroup); } return $form; } }, _generateTable: { value: function generateTable(rows, columns) { rows = parseInt(rows) + 1; columns = parseInt(rows); if(!columns) columns = 1; var content = '<table class="table">'; for(var row = 1; row <= rows; row++) { if (row == 1) content += '<thead>'; else if (row == ((rows - row) - 1)) content += '<tbody>'; content += '<tr>'; for(var column = 1; column <= columns; column++) { if (row == 1) content += '<th>Header ' + column + '</th>'; else content += '<td> </td>'; } content += '</tr>'; if (row == 1) content += '</thead>'; else if (row == rows) content += '</tbody>'; } content += '</table>'; return content; } }, _updateState: { value: function updateState($target, reset) { var _this = this; if (_this._config.mode == 'editor') { var statInfo = _this._getTextStat(_this._$content.get(0)); var pathInfo = _this._getPath($target, _this._$content, false); if(!reset) { switch (pathInfo['tags'][0]) { case 'b' : _this._$toolbar.find('[data-action="text"]').removeClass('active'); _this._$toolbar.find('[data-action="text"][data-value="bold"]').addClass('active'); break; case 'u' : _this._$toolbar.find('[data-action="text"]').removeClass('active'); _this._$toolbar.find('[data-action="text"][data-value="underline"]').addClass('active'); break; case 'sub' : _this._$toolbar.find('[data-action="text"]').removeClass('active'); _this._$toolbar.find('[data-action="text"][data-value="subscript"]').addClass('active'); break; case 'sup' : _this._$toolbar.find('[data-action="text"]').removeClass('active'); _this._$toolbar.find('[data-action="text"][data-value="superscript"]').addClass('active'); break; case 'i' : _this._$toolbar.find('[data-action="text"]').removeClass('active'); _this._$toolbar.find('[data-action="text"][data-value="italic"]').addClass('active'); break; case 'a' : _this._$toolbar.find('[data-action="insert"]').removeClass('active'); _this._$toolbar.find('[data-action="insert"][data-value="link"]').addClass('active'); break; case 'p' : _this._$toolbar.find('[data-action="align"]').removeClass('active'); if ($target.css('text-align') == 'center') _this._$toolbar.find('[data-action="align"][data-value="center"]').addClass('active'); else if ($target.css('text-align') == 'right') _this._$toolbar.find('[data-action="align"][data-value="right"]').addClass('active'); else if ($target.css('text-align') == 'justify') _this._$toolbar.find('[data-action="align"][data-value="justify"]').addClass('active'); else _this._$toolbar.find('[data-action="align"][data-value="left"]').addClass('active'); break; default : _this._$toolbar.find('[data-action="text"]').removeClass('active'); _this._$toolbar.find('[data-action="align"]').removeClass('active'); _this._$toolbar.find('[data-action="insert"]').removeClass('active'); break; } } else { _this._$toolbar.find('button[data-action]').removeClass('active'); } _this._$statusbar.path.text(pathInfo['path']); _this._$statusbar.stat.text('Length: ' + statInfo['length'] + ', chars: ' + statInfo['chars'] + ', words: ' + statInfo['words']); } else { var position = _this._getTextPosition(_this._$content.get(0)); _this._$statusbar.path.empty(); if(parseInt(position['selected']) > 0) _this._$statusbar.stat.text('Line: ' + position['line'] + ', column: ' + position['end'] + ', selected: ' + position['selected']); else _this._$statusbar.stat.text('Line: ' + position['line'] + ', column: ' + position['end']); } return _this._$element; } }, }, { Default: { get: function() { return defaults; } }, _jQueryInterface: { value: function _jQueryInterface(config) { let _this = this; config = config || {}; if (/destroy|hide/.test(config)) { let element = $(_this); element.removeClass('hide'); let editor = element.parent('.wysiwyg-editor'); editor.replaceWith(element); return; } return _this.each(function() { let $this = $(_this); let _config = $.extend({}, WYSIWYG.Default, $this.data(), typeof config === "object" && config); new Editor(_this, _config); }); } } }); return Editor; })(); $.fn[className] = Editor._jQueryInterface; $.fn[className].Constructor = Editor; $.fn[className].noConflict = function() { $.fn[className] = _jQueryNoConflict; return Editor._jQueryInterface; }; return Editor; })(jQuery); }(jQuery); //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["wysiwyg.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"wysiwyg.js","sourcesContent":["/**\n * Simple WYSIWYG editor for Bootstrap3\n *\n * @category        jQuery Plugin\n * @version         1.1.4\n * @author          Alexsander Vyshnyvetskyy <alex.vyshnyvetskyy@gmail.com>\n * @link            http://wdmg.github.io/bootstrap-wysiwyg\n * @copyright       Copyright (c) 2019 - 2020 W.D.M.Group, Ukraine\n * @license         https://opensource.org/licenses/MIT Massachusetts Institute of Technology (MIT) License\n *\n */\n\n+function($) {\n\n    \"use strict\";\n    var _createClass = (function() {\n        function defineProperties(target, props) {\n            for (var key in props) {\n                var prop = props[key];\n                prop.configurable = true;\n                if (prop.value) prop.writable = true;\n            }\n            Object.defineProperties(target, props);\n        };\n        return function(Constructor, protoProps, staticProps) {\n            if (protoProps) defineProperties(Constructor.prototype, protoProps);\n            if (staticProps) defineProperties(Constructor, staticProps);\n            return Constructor;\n        };\n    })();\n\n    var _classCallCheck = function(instance, Constructor) {\n        if (!(instance instanceof Constructor)) {\n            throw new TypeError(\"Cannot call a class as a function\");\n        }\n    };\n\n    var WYSIWYG = (function($) {\n\n        var className = \"wysiwyg\";\n        var _jQueryNoConflict = $.fn[className];\n        var defaults = {\n            toolbar: [\n                ['mode'],\n                ['operations', ['undo', 'rendo', 'cut', 'copy', 'paste']],\n                ['styles'],\n                ['fonts', ['select', 'size']],\n                ['text', ['bold', 'italic', 'underline', 'strike', 'subscript', 'superscript', 'font-color', 'bg-color']],\n                ['align', ['left', 'center', 'right', 'justify']],\n                ['lists', ['unordered', 'ordered', 'indent', 'outdent']],\n                ['components', ['table', /*'chart'*/]],\n                ['intervals', ['line-height', 'letter-spacing']],\n                ['insert', ['emoji', 'link', 'image', 'video', 'symbol', /*'bookmark'*/]],\n                ['special', ['print', 'unformat', 'visual', 'clean']],\n                /*['fullscreen'],*/\n            ],\n            fontSizes: ['8px', '9px', '10px', '11px', '12px', '14px', '15px', '16px', '18px', '20px', '24px', '30px', '32px', '36px', '48px'],\n            fontSizeDefault: '12px',\n            fontFamilies: ['Open Sans', 'Arial', 'Arial Black', 'Courier', 'Courier New', 'Comic Sans MS', 'Helvetica', 'Impact', 'Lucida Grande', 'Lucida Sans', 'Tahoma', 'Times', 'Times New Roman', 'Verdana'],\n            fontFamilyDefault: 'Open Sans',\n            emojiDefault: [\"\\u{1f600}\", \"\\u{1f62c}\", \"\\u{1f601}\", \"\\u{1f602}\", \"\\u{1f603}\", \"\\u{1f604}\", \"\\u{1f605}\", \"\\u{1f606}\", \"\\u{1f607}\", \"\\u{1f609}\", \"\\u{1f60a}\", \"\\u{1f642}\", \"\\u{1f643}\", \"\\u{1f60b}\", \"\\u{1f60c}\", \"\\u{1f60d}\", \"\\u{1f618}\", \"\\u{1f617}\", \"\\u{1f619}\", \"\\u{1f61a}\", \"\\u{1f61c}\", \"\\u{1f61d}\", \"\\u{1f61b}\", \"\\u{1f911}\", \"\\u{1f913}\", \"\\u{1f60e}\", \"\\u{1f917}\", \"\\u{1f60f}\", \"\\u{1f636}\", \"\\u{1f610}\", \"\\u{1f611}\", \"\\u{1f612}\", \"\\u{1f644}\", \"\\u{1f914}\", \"\\u{1f633}\", \"\\u{1f61e}\", \"\\u{1f61f}\", \"\\u{1f620}\", \"\\u{1f621}\", \"\\u{1f614}\", \"\\u{1f615}\", \"\\u{1f641}\", \"\\u{1f623}\", \"\\u{1f616}\", \"\\u{1f62b}\", \"\\u{1f629}\", \"\\u{1f624}\", \"\\u{1f62e}\", \"\\u{1f631}\", \"\\u{1f628}\", \"\\u{1f630}\", \"\\u{1f62f}\", \"\\u{1f626}\", \"\\u{1f627}\", \"\\u{1f622}\", \"\\u{1f625}\", \"\\u{1f62a}\", \"\\u{1f613}\", \"\\u{1f62d}\", \"\\u{1f635}\", \"\\u{1f632}\", \"\\u{1f910}\", \"\\u{1f637}\", \"\\u{1f912}\", \"\\u{1f915}\", \"\\u{1f634}\", \"\\u{1f4a4}\"],\n            symbolsDefault: [\"&lt;\", \"&gt;\", \"&laquo;\", \"&raquo;\", \"&lsaquo;\", \"&rsaquo;\", \"&quot;\", \"&prime;\", \"&Prime;\", \"&lsquo;\", \"&rsquo;\", \"&sbquo;\", \"&ldquo;\", \"&rdquo;\", \"&bdquo;\", \"&#10076;\", \"&#10075;\", \"&amp;\", \"&apos;\", \"&sect;\", \"&copy;\", \"&not;\", \"&reg;\", \"&macr;\", \"&deg;\", \"&plusmn;\", \"&sup1;\", \"&sup2;\", \"&sup3;\", \"&frac14;\", \"&frac12;\", \"&frac34;\", \"&acute;\", \"&micro;\", \"&para;\", \"&middot;\", \"&iquest;\", \"&fnof;\", \"&trade;\", \"&bull;\", \"&hellip;\", \"&oline;\", \"&ndash;\", \"&mdash;\", \"&permil;\", \"&#125;\", \"&#123;\", \"&#61;\", \"&ne;\", \"&cong;\", \"&asymp;\", \"&le;\", \"&ge;\", \"&ang;\", \"&perp;\", \"&radic;\", \"&sum;\", \"&int;\", \"&#8251;\", \"&divide;\", \"&infin;\", \"&#64;\", \"&#91;\", \"&#93;\", \"&larr;\", \"&uarr;\", \"&rarr;\", \"&darr;\", \"&harr;\", \"&crarr;\", \"&lArr;\", \"&uArr;\", \"&rArr;\", \"&dArr;\", \"&hArr;\", \"&#10144;\", \"&#10148;\", \"&#10149;\", \"&#10150;\", \"&#10163;\", \"&#8634;\", \"&#8635;\", \"&#8679;\", \"&#8617;\", \"&#11015;\", \"&#11014;\", \"&spades;\", \"&clubs;\", \"&hearts;\", \"&diams;\", \"&#9825;\", \"&#9826;\", \"&#9828;\", \"&#9831;\", \"&#8372;\", \"&euro;\", \"&dollar;\", \"&cent;\", \"&pound;\", \"&#8381;\", \"&yen;\", \"&#8377;\", \"&#22291;\", \"&#8376;\"],\n            colorPalette: [[\"rgb(0, 0, 0)\",\"rgb(67, 67, 67)\",\"rgb(102, 102, 102)\",\"rgb(153, 153, 153)\",\"rgb(183, 183, 183)\",\"rgb(204, 204, 204)\",\"rgb(217, 217, 217)\",\"rgb(239, 239, 239)\",\"rgb(243, 243, 243)\",\"rgb(255, 255, 255)\"],[\"rgb(152, 0, 0)\",\"rgb(255, 0, 0)\",\"rgb(255, 153, 0)\",\"rgb(255, 255, 0)\",\"rgb(0, 255, 0)\",\"rgb(0, 255, 255)\",\"rgb(74, 134, 232)\",\"rgb(0, 0, 255)\",\"rgb(153, 0, 255)\",\"rgb(255, 0, 255)\"],[\"rgb(230, 184, 175)\",\"rgb(244, 204, 204)\",\"rgb(252, 229, 205)\",\"rgb(255, 242, 204)\",\"rgb(217, 234, 211)\",\"rgb(208, 224, 227)\",\"rgb(201, 218, 248)\",\"rgb(207, 226, 243)\",\"rgb(217, 210, 233)\",\"rgb(234, 209, 220)\",\"rgb(221, 126, 107)\",\"rgb(234, 153, 153)\",\"rgb(249, 203, 156)\",\"rgb(255, 229, 153)\",\"rgb(182, 215, 168)\",\"rgb(162, 196, 201)\",\"rgb(164, 194, 244)\",\"rgb(159, 197, 232)\",\"rgb(180, 167, 214)\",\"rgb(213, 166, 189)\",\"rgb(204, 65, 37)\",\"rgb(224, 102, 102)\",\"rgb(246, 178, 107)\",\"rgb(255, 217, 102)\",\"rgb(147, 196, 125)\",\"rgb(118, 165, 175)\",\"rgb(109, 158, 235)\",\"rgb(111, 168, 220)\",\"rgb(142, 124, 195)\",\"rgb(194, 123, 160)\",\"rgb(166, 28, 0)\",\"rgb(204, 0, 0)\",\"rgb(230, 145, 56)\",\"rgb(241, 194, 50)\",\"rgb(106, 168, 79)\",\"rgb(69, 129, 142)\",\"rgb(60, 120, 216)\",\"rgb(61, 133, 198)\",\"rgb(103, 78, 167)\",\"rgb(166, 77, 121)\",\"rgb(133, 32, 12)\",\"rgb(153, 0, 0)\",\"rgb(180, 95, 6)\",\"rgb(191, 144, 0)\",\"rgb(56, 118, 29)\",\"rgb(19, 79, 92)\",\"rgb(17, 85, 204)\",\"rgb(11, 83, 148)\",\"rgb(53, 28, 117)\",\"rgb(116, 27, 71)\",\"rgb(91, 15, 0)\",\"rgb(102, 0, 0)\",\"rgb(120, 63, 4)\",\"rgb(127, 96, 0)\",\"rgb(39, 78, 19)\",\"rgb(12, 52, 61)\",\"rgb(28, 69, 135)\",\"rgb(7, 55, 99)\",\"rgb(32, 18, 77)\",\"rgb(76, 17, 48)\"]],\n            mode: 'editor',\n            language: 'en-us',\n            translations: {},\n            highlight: true,\n            debug: false\n        };\n\n        const Styles = {\n            'Header H1': {\n                'action': 'formatblock',\n                'value': 'h1',\n                'wrap': '<h1 />',\n            },\n            'Header H2': {\n                'action': 'formatblock',\n                'value': 'h2',\n                'wrap': '<h2 />',\n            },\n            'Header H3': {\n                'action': 'formatblock',\n                'value': 'h3',\n                'wrap': '<h3 />',\n            },\n            'Header H4': {\n                'action': 'formatblock',\n                'value': 'h4',\n                'wrap': '<h4 />',\n            },\n            'Header H5': {\n                'action': 'formatblock',\n                'value': 'h5',\n                'wrap': '<h5 />',\n            },\n            'Header H6': {\n                'action': 'formatblock',\n                'value': 'h6',\n                'wrap': '<h6 />',\n            },\n            'Paragraph': {\n                'action': 'formatblock',\n                'value': 'p',\n                'wrap': '<p />',\n            },\n            'Blockquote': {\n                'action': 'formatblock',\n                'value': 'blockquote',\n                'wrap': '<blockquote />',\n            },\n            'Preformatted': {\n                'action': 'formatblock',\n                'value': 'pre',\n                'wrap': '<pre />',\n            },\n            'Div block': {\n                'action': 'formatblock',\n                'value': 'div',\n                'wrap': '<div />',\n            }\n        }\n\n        const videoServices = {\n            youtube: 'YouTube',\n            vimeo: 'Vimeo',\n            dailymotion: 'Dailymotion',\n            /*hulu: 'Hulu',\n            twitch: 'Twitch',\n            facebook: 'Facebook',\n            vkontakte: 'vKontakte',\n            twitter: 'Twitter',\n            ustream: 'Ustream',*/\n            source: 'Source media',\n            /*embed: 'Embed code'*/\n        };\n\n        const urlSchemes = {\n            https: 'https://',\n            http: 'http://',\n            mailto: 'mailto://',\n            ftp: 'ftp://',\n            feed: 'feed://',\n            news: 'news://',\n            tel: 'tel:',\n            skype: 'skype:',\n            telegram: 'tg://',\n            whatsapp: 'whatsapp:',\n            viber: 'viber:',\n            other: 'other'\n        };\n\n        const urlLinkTarget = {\n            blank: 'New tab',\n            top: 'Main tab',\n            self: 'Current tab',\n            parent: 'Parent tab',\n            /*iframe: 'Iframe',\n            popup: 'PopUp',*/\n        };\n\n        const urlLinkRel = {\n            nofollow: 'Do not follow (for robots)',\n            noreferrer: 'Do not pass HTTP-referrer',\n            /*archives: 'Link to the site archive',\n            author: 'Link to the page about the author on the same domain',\n            bookmark: 'Permalink to a section or post',\n            first: 'Link to the first page',\n            help: 'Help document link',\n            index: 'Content Link',\n            last: 'Link to the last page',\n            license: 'Link to page with license agreement or copyright',\n            me: 'Link to the author’s page on another domain',\n            next: 'Link to the next page or section',\n            prefetch: 'Indicates that the specified resource must be cached in advance.',\n            prev: 'Link to previous page or section',\n            search: 'Search Link',\n            sidebar: 'Add link to browser favorites',\n            tag: 'Indicates that the label (tag) is related to the current document.',\n            up: 'Link to the parent page',\n            answer: 'Answer to the question',\n            chapter: 'Section or chapter of the current document',\n            co-worker: 'Link to colleague’s page',\n            colleague: 'Link to colleague’s page (not for work)',\n            contact: 'Link to the page with contact information',\n            details: 'Link to the page with details',\n            edit: 'Editable version of the current document',\n            friend: 'Link to friend’s page',\n            question: 'Link to the question page',*/\n        };\n\n        var Editor = (function() {\n\n            function Editor($element, config) {\n                var _this = this;\n                _classCallCheck(_this, Editor);\n\n                // Merge default and custom options\n                _this._config = $.extend({}, defaults, config);\n\n                if (_this._config.debug)\n                    console.log('Init WYSIWYG editor...');\n\n                // Configure variables\n                _this._editorId = 'wysiwyg-' + (String.fromCharCode(Math.floor(Math.random() * 11)) + Math.floor(Math.random() * 1000000)).trim();\n                _this._$element = $element instanceof jQuery ? $element : $($element);\n                _this._inputId = _this._$element.attr('id');\n\n                // Wrap text input to container\n                _this._$editor = $('<div id=\"' + _this._editorId + '\" aria-describedby=\"#' + _this._inputId + '\" class=\"wysiwyg-editor\" />');\n                _this._$element.wrap(_this._$editor);\n\n                // Add content to editor\n                _this._$content = $('<div class=\"editor-content\" contenteditable=\"true\" />');\n                _this._$content.html(_this._$element.val());\n                _this._$element.before(_this._$content);\n                _this._source = _this._$element.val();\n\n                _this._selection = document.getSelection();\n                _this._popoverIsVisible = false;\n                _this._$lastFocus = null;\n\n                // Add toolbar to editor\n                _this._$toolbar = $('<div class=\"wysiwyg-toolbar btn-toolbar\" />');\n                _this._$content.before(_this._$toolbar);\n\n                // Add statusbar to editor\n                _this._$statusbar = $('<div class=\"editor-statusbar\" />');\n                _this._$statusbar.stat = $('<span class=\"editor-statusbar-stat\" />');\n                _this._$statusbar.path = $('<span class=\"editor-statusbar-path\" />');\n                _this._$statusbar.append(_this._$statusbar.stat);\n                _this._$statusbar.append(_this._$statusbar.path);\n                _this._$content.after(_this._$statusbar);\n\n                // Hide input editor\n                _this._$element.addClass('hide');\n\n                // Build toolbar by config\n                if(typeof (_this._config.toolbar) == 'object') {\n                    $.each(_this._config.toolbar, function (index, elem) {\n\n                        var $toolbar = $('<div id=\"toolbarGroup-' + elem[0] + '\" class=\"btn-group\" role=\"group\" />');\n\n                        if(elem[0] === 'mode') { // Editor mode switcher\n\n                            var editorButton = _this._buildTollbarButton('mode', 'editor', \"fa fa-eye\", null, \"Editor\");\n                            var sourceButton = _this._buildTollbarButton('mode', 'source', \"fa fa-code\", null, \"Source\");\n\n                            if(_this._config.mode == 'editor')\n                                editorButton.addClass('active');\n                            else\n                                sourceButton.addClass('active');\n\n                            $toolbar.append(editorButton);\n                            $toolbar.append(sourceButton);\n\n                        } else if(elem[0] === 'operations') { // Operations editor controls\n\n                            $toolbar.append(_this._buildTollbarButton('operations', 'undo', \"fa fa-reply\", null, \"Undo\"));\n                            $toolbar.append(_this._buildTollbarButton('operations', 'rendo', \"fa fa-share\", null, \"Rendo\"));\n                            $toolbar.append(_this._buildTollbarButton('operations', 'cut', \"fa fa-cut\", null, \"Cut\"));\n                            $toolbar.append(_this._buildTollbarButton('operations', 'copy', \"fa fa-copy\", null, \"Copy\"));\n                            $toolbar.append(_this._buildTollbarButton('operations', 'paste', \"fa fa-clipboard\", null, \"Paste\"));\n\n                        } else if(elem[0] === 'styles') { // Editor mode switcher\n\n                            $toolbar.append(_this._buildTollbarDropdown('select-style', Styles, \"Paragraph\", \"Text style\"));\n\n                        } else if(elem[0] === 'fonts') { // Font select and size\n\n                            if(elem[1].indexOf('select', 0) !== -1) {\n\n                                var fonts = {};\n                                $.each(_this._config.fontFamilies, function(index, value) {\n                                    fonts[value] = {\n                                        'action': 'fontname',\n                                        'value': value,\n                                        'style': \"font-family: \" + value + \";\"\n                                    };\n                                });\n\n                                $toolbar.append(_this._buildTollbarDropdown('font-select', fonts, _this._config.fontFamilyDefault, \"Font family\"));\n                            }\n\n                            if(elem[1].indexOf('size', 0) !== -1) {\n                                var sizes = {};\n                                $.each(_this._config.fontSizes, function(index, value) {\n                                    sizes[value] = {\n                                        'action': 'fontsize',\n                                        'value': value\n                                    };\n                                });\n                                $toolbar.append(_this._buildTollbarDropdown('font-size', sizes, _this._config.fontSizeDefault, \"Font size\"));\n                            }\n\n                        } else if(elem[0] === 'text') { // Text decoration\n\n                            if(elem[1].indexOf('bold', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'bold', \"fa fa-bold\", null, \"Bold\"));\n\n                            if(elem[1].indexOf('italic', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'italic', \"fa fa-italic\", null, \"Italic\"));\n\n                            if(elem[1].indexOf('underline', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'underline', \"fa fa-underline\", null, \"Underline\"));\n\n                            if(elem[1].indexOf('strike', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'strike', \"fa fa-strikethrough\", null, \"Striked text\"));\n\n                            if(elem[1].indexOf('subscript', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'subscript', \"fa fa-subscript\", null, \"Subscript\"));\n\n                            if(elem[1].indexOf('superscript', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'superscript', \"fa fa-superscript\", null, \"Superscript\"));\n\n                            if(elem[1].indexOf('bg-color', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'font-color', \"fa fa-font\", null, \"Font color\", _this._buildColorPalette(_this._config.colorPalette, \"font-color\", null)));\n\n                            if(elem[1].indexOf('bg-color', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('text', 'bg-color', \"fa fa-paint-brush\", null, \"Background color\", _this._buildColorPalette(_this._config.colorPalette, \"bg-color\", true)));\n\n                        } else if(elem[0] === 'align') { // Text aligment\n\n                            if(elem[1].indexOf('left', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('align', 'left', \"fa fa-align-left\", null, \"Align left\", null));\n\n                            if(elem[1].indexOf('center', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('align', 'center', \"fa fa-align-center\", null, \"Align center\", null));\n\n                            if(elem[1].indexOf('right', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('align', 'right', \"fa fa-align-right\", null, \"Align right\", null));\n\n                            if(elem[1].indexOf('justify', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('align', 'justify', \"fa fa-align-justify\", null, \"Justify content\", null));\n\n                        } else if(elem[0] === 'lists') { // Lists && outdent\n\n                            if(elem[1].indexOf('unordered', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('lists', 'unordered', \"fa fa-list-ul\", null, \"Unordered list\"));\n\n                            if(elem[1].indexOf('ordered', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('lists', 'ordered', \"fa fa-list-ol\", null, \"Ordered list\"));\n\n                            if(elem[1].indexOf('indent', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('lists', 'indent', \"fa fa-indent\", null, \"Indent\"));\n\n                            if(elem[1].indexOf('outdent', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('lists', 'outdent', \"fa fa-outdent\", null, \"Outdent\"));\n\n                        } else if(elem[0] === 'components') { // Components\n\n                            if(elem[1].indexOf('table', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('components', 'table', \"fa fa-table\", null, \"Insert table\", _this._buildTableGrid()));\n\n                            if(elem[1].indexOf('chart', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('components', 'chart', \"fa fa-pie-chart\", null, \"Add chart\"));\n\n                        } else if(elem[0] === 'intervals') { // Text properties\n\n                            if(elem[1].indexOf('line-height', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('interval', 'line-height', \"fa fa-text-height\", null, \"Lines interval\", _this._buildLineHeightList()));\n\n                            if(elem[1].indexOf('letter-spacing', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('interval', 'letter-spacing', \"fa fa-text-width\", null, \"Letter spacing\", _this._buildLetterSpacingList()));\n\n                        } else if(elem[0] === 'insert') { // Inserts\n\n                            if(elem[1].indexOf('emoji', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('insert', 'emoji', \"fa fa-smile fa-smile-o\", null, \"Add emoji\", _this._buildEmojiList()));\n\n                            if(elem[1].indexOf('link', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('insert', 'link', \"fa fa-link\", null, \"Add URL\", _this._buildUrlForm('link')));\n\n                            if(elem[1].indexOf('image', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('insert', 'image', \"fa fa-image\", null, \"Add image\", _this._buildUrlForm('image')));\n\n                            if(elem[1].indexOf('video', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('insert', 'video', \"fa fa-video-camera fa-video\", null, \"Add video\", _this._buildUrlForm('video')));\n\n                            if(elem[1].indexOf('symbol', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('insert', 'symbol', \"fa fa-hashtag\", null, \"Add symbol\", _this._buildSymbolsList()));\n\n                            if(elem[1].indexOf('bookmark', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('insert', 'bookmark', \"fa fa-bookmark\", null, \"Add bookmark\"));\n\n                        } else if(elem[0] === 'special') { // Inserts\n\n                            if(elem[1].indexOf('print', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('special', 'print', \"fa fa-print\", null, \"Print\"));\n\n                            if(elem[1].indexOf('clean', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('special', 'clean', \"fa fa-eraser\", null, \"Erase style\"));\n\n                            if(elem[1].indexOf('visual', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('special', 'visual', \"fa fa-solar-panel\", null, \"Visual blocks\"));\n\n                            if(elem[1].indexOf('unformat', 0) !== -1)\n                                $toolbar.append(_this._buildTollbarButton('special', 'unformat', \"fa fa-trash-o fa-trash-alt\", null, \"Clear HTML\"));\n\n                        } else if(elem[0] === 'fullscreen') { // Fullscreen mode\n\n                            $toolbar.addClass('pull-right');\n                            $toolbar.append(_this._buildTollbarButton('fullscreen', true, \"fa fa-arrows-alt\", null, \"Fullscreen mode\"));\n\n                        }\n\n                        _this._$toolbar.append($toolbar);\n\n                    });\n                }\n\n                // Set behavior for toolbar buttons\n                if(_this._$toolbar.length) {\n                    _this._$toolbar.on('click', '[data-action]', function(event) {\n                        var $target = $(event.currentTarget);\n                        var action = $target.data('action');\n                        var selection = _this._selection;\n                        var value = $target.data('value');\n                        var data = $target.data();\n\n                        if (typeof (action) !== 'undefined' && typeof (value) !== 'undefined') {\n\n                            if (_this._config.debug)\n                                console.log('Switch action: `' + action + '` with value: `' + value + '`');\n\n                            switch (action) {\n\n                                case 'mode':\n                                    switch (value) {\n                                        case 'editor':\n                                            if (_this._config.mode !== value) {\n                                                _this._config.mode = value;\n                                                _this._$content.html(_this._source);\n                                                _this._$toolbar.find('[data-action=\"mode\"]').removeClass('active');\n                                                _this._$toolbar.find('[data-action=\"mode\"][data-value=\"editor\"]').addClass('active');\n                                                _this._$content.addClass('editor-mode').removeClass('source-mode');\n                                                _this._$content.focus();\n                                            }\n\n                                            _this._$toolbar.find('.btn-group').removeClass('hide');\n                                            break;\n\n                                        case 'source':\n\n                                            _this._$toolbar.find('.btn-group').not('#toolbarGroup-' + action).addClass('hide');\n\n                                            if (_this._config.mode !== value) {\n\n                                                _this._config.mode = value;\n                                                _this._source = _this._$content.html();\n\n                                                var $source = $('<pre />');\n                                                $source.text(_this._source);\n\n                                                _this._$content.html(_this._trimSource($source.html()));\n\n                                                if (_this._config.highlight) {\n                                                    hljs.initHighlighting.called = false;\n                                                    hljs.configure({\n                                                        useBR: true,\n                                                        languages: ['html', 'javascript', 'css']\n                                                    });\n                                                    hljs.highlightBlock(_this._$content.get(0));\n                                                }\n\n                                                _this._$toolbar.find('[data-action=\"mode\"]').removeClass('active');\n                                                _this._$toolbar.find('[data-action=\"mode\"][data-value=\"source\"]').addClass('active');\n\n                                                _this._$content.removeClass('editor-mode').addClass('source-mode');\n                                                _this._$content.focus();\n                                            }\n                                            break;\n                                    }\n                                    break;\n\n                                case 'formatblock':\n                                    _this._formatDoc('formatblock', value);\n                                    break;\n\n                                case 'fontname':\n                                    _this._formatDoc('fontname', value);\n                                    break;\n\n                                case 'fontsize':\n                                    _this._selection.anchorNode.parentElement.removeAttribute(\"size\");\n                                    _this._selection.anchorNode.parentElement.style.fontSize = value;\n                                    break;\n\n                                case 'style':\n                                    var styles = _this._selection.anchorNode.parentElement.style.cssText;\n\n                                    if(styles)\n                                        styles += value;\n                                    else\n                                        styles = value;\n\n                                    _this._selection.anchorNode.parentElement.removeAttribute(\"style\");\n                                    _this._selection.anchorNode.parentElement.style = styles;\n                                    break;\n\n                                case 'fullscreen':\n\n                                    if (_this._config.debug)\n                                        console.log('Fire action: ' + action + ' with value: ' + value + ' is not supported.');\n\n                                    break;\n\n                                case 'operations':\n                                    switch (value) {\n                                        case 'undo':\n                                            _this._formatDoc('undo');\n                                            break;\n\n                                        case 'rendo':\n                                            _this._formatDoc('rendo');\n                                            break;\n                                        case 'cut':\n                                            _this._formatDoc('cut');\n                                            break;\n\n                                        case 'copy':\n                                            _this._formatDoc('copy');\n                                            break;\n\n                                        case 'paste':\n                                            _this._formatDoc('paste');\n                                            break;\n                                    }\n                                    break;\n\n                                case 'text':\n                                    switch (value) {\n                                        case 'bold':\n                                            _this._formatDoc('bold');\n                                            break;\n\n                                        case 'italic':\n                                            _this._formatDoc('italic');\n                                            break;\n\n                                        case 'underline':\n                                            _this._formatDoc('underline');\n                                            break;\n\n                                        case 'strike':\n                                            _this._formatDoc('strikeThrough');\n                                            break;\n\n                                        case 'subscript':\n                                            _this._formatDoc('subscript');\n                                            break;\n\n                                        case 'superscript':\n                                            _this._formatDoc('superscript');\n                                            break;\n\n                                    }\n                                    break;\n\n                                case 'font-color':\n                                    if(value == 'unset') {\n\n                                        if(_this._selection.anchorNode)\n                                            _this._selection.anchorNode.parentElement.style.backgroundColor = \"\";\n\n                                        if(_this._selection.anchorNode.parentElement.style.length)\n                                            _this._selection.anchorNode.parentElement.removeAttribute(\"style\");\n\n                                    } else {\n                                        _this._$toolbar.find('[data-action=\"text\"][data-value=\"font-color\"] > span').css('border-bottom-color', value);\n                                        _this._formatDoc('foreColor', value);\n                                    }\n                                    break;\n\n                                case 'bg-color':\n                                    if(value == 'unset') {\n\n                                        if(_this._selection.anchorNode)\n                                            _this._selection.anchorNode.parentElement.style.backgroundColor = \"\";\n\n                                        if(_this._selection.anchorNode.parentElement.style.length)\n                                            _this._selection.anchorNode.parentElement.removeAttribute(\"style\");\n\n                                    } else {\n                                        _this._$toolbar.find('[data-action=\"text\"][data-value=\"bg-color\"] > span').css('border-bottom-color', value);\n                                        _this._formatDoc('hiliteColor', value);\n                                    }\n                                    break;\n\n                                case 'align':\n                                    switch (value) {\n                                        case 'left':\n                                            _this._formatDoc('justifyLeft');\n                                            break;\n\n                                        case 'center':\n                                            _this._formatDoc('justifyCenter');\n                                            break;\n\n                                        case 'right':\n                                            _this._formatDoc('justifyRight');\n                                            break;\n\n                                        case 'justify':\n                                            _this._formatDoc('justifyFull');\n                                            break;\n                                    }\n                                    break;\n\n\n                                case 'lists':\n                                    switch (value) {\n                                        case 'unordered':\n                                            _this._formatDoc('insertUnorderedList');\n                                            break;\n\n                                        case 'ordered':\n                                            _this._formatDoc('insertOrderedList');\n                                            break;\n\n                                        case 'indent':\n                                            _this._formatDoc('indent');\n                                            break;\n\n                                        case 'outdent':\n                                            _this._formatDoc('outdent');\n                                            break;\n                                    }\n                                    break;\n\n                                case 'insert-table':\n                                    if(_this._selection.anchorNode) {\n                                        var options = value.split('|', 2);\n                                        var $parent = $(_this._selection.anchorNode.parentElement);\n                                        var content = _this._generateTable(parseFloat(options[0]), parseFloat(options[2]));\n                                        $parent.after(content);\n                                    }\n                                    break;\n\n                                case 'components':\n                                    switch (value) {\n\n                                        case 'chart':\n                                            if (_this._config.debug)\n                                                console.log('Fire action: ' + action + ' with value: ' + value + ' is not supported.');\n\n                                            break;\n                                    }\n                                    break;\n\n                                case 'line-height':\n                                    if(_this._selection.anchorNode) {\n                                        var lineHeight = parseFloat(value) * 100 + \"%\";\n\n                                        if (parseFloat(value) == 0)\n                                            _this._selection.anchorNode.parentElement.style.lineHeight = \"inherit\";\n                                        else\n                                            _this._selection.anchorNode.parentElement.style.lineHeight = lineHeight;\n                                    }\n                                    break;\n\n                                case 'letter-spacing':\n                                    if(_this._selection.anchorNode) {\n                                        var letterSpacing = parseFloat(value) + \"px\";\n\n                                        if (parseFloat(value) == 0)\n                                            _this._selection.anchorNode.parentElement.style.letterSpacing = \"inherit\";\n                                        else\n                                            _this._selection.anchorNode.parentElement.style.letterSpacing = letterSpacing;\n\n                                    }\n                                    break;\n\n                                case 'add-url':\n                                    var text = _this._selection.toString();\n                                    if(_this._selection && text) {\n\n                                        var url = value;\n                                        console.log(data);\n\n                                        if (data.scheme) {\n                                            if (data.scheme == 'https')\n                                                url = 'https://' + url;\n                                            else if (data.scheme == 'http')\n                                                url = 'http://' + url;\n                                            else if (data.scheme == 'mailto')\n                                                url = 'mailto://' + url;\n                                            else if (data.scheme == 'ftp')\n                                                url = 'ftp://' + url;\n                                            else if (data.scheme == 'feed')\n                                                url = 'feed://' + url;\n                                            else if (data.scheme == 'news')\n                                                url = 'news://' + url;\n                                            else if (data.scheme == 'tel')\n                                                url = 'tel:' + url;\n                                            else if (data.scheme == 'skype')\n                                                url = 'skype:' + url;\n                                            else if (data.scheme == 'tg')\n                                                url = 'tg://' + url;\n                                            else if (data.scheme == 'whatsapp')\n                                                url = 'whatsapp:' + url;\n                                            else if (data.scheme == 'viber')\n                                                url = 'viber:' + url;\n                                        }\n\n                                        var title = '';\n                                        if (data.title) {\n                                            title = ' title=\"' + data.title + '\"';\n                                        }\n\n                                        var className = '';\n                                        if (data.class) {\n                                            className = ' class=\"' + data.class + '\"';\n                                        }\n\n                                        var target = '';\n                                        if (data.target) {\n                                            if (data.target == 'blank')\n                                                target = ' target=\"_blank\"';\n                                            else if (data.target == 'top')\n                                                target = ' target=\"_top\"';\n                                            else if (data.target == 'self')\n                                                target = ' target=\"_self\"';\n                                            else if (data.target == 'parent')\n                                                target = ' target=\"_parent\"';\n                                        }\n\n                                        var rel = '';\n                                        if (data.relation) {\n                                            if (data.relation == 'nofollow')\n                                                rel = ' rel=\"nofollow\"';\n                                            else if (data.relation == 'noreferrer')\n                                                rel = ' rel=\"noreferrer\"';\n                                        }\n\n                                        var $link = $('<a href=\"' + url + '\"' + title + className + target + rel + ' />');\n                                        $link.text(text);\n\n                                        var range = _this._selection.getRangeAt(0);\n                                        range.deleteContents();\n                                        range.insertNode($link.get(0));\n                                    }\n                                    break;\n\n                                case 'add-video':\n                                    if(_this._selection && value) {\n\n                                        var url = value;\n                                        if (data.service) {\n                                            if (data.service == 'youtube') {\n                                                var videoId = null;\n                                                var regExp = /^.*(youtube\\/|youtu.be\\/|v\\/|u\\/\\w\\/|embed\\/|watch\\?v=|\\&v=)([^#\\&\\?]*).*/;\n                                                var match = url.match(regExp);\n\n                                                if (match !== null && match[2].length == 11)\n                                                    videoId = match[2];\n\n                                                if (videoId)\n                                                    url = 'https://www.youtube.com/embed/' + videoId;\n\n                                            } else if (data.service == 'vimeo') {\n                                                var videoId = null;\n                                                var regExp = /(?:www\\.|player\\.)?vimeo.com\\/(?:channels\\/(?:\\w+\\/)?|groups\\/(?:[^\\/]*)\\/videos\\/|album\\/(?:\\d+)\\/video\\/|video\\/|)(\\d+)(?:[a-zA-Z0-9_\\-]+)?/i;\n                                                var match = url.match(regExp);\n\n                                                if (match !== null && match[1].length == 9)\n                                                    videoId = match[1];\n\n                                                if (videoId)\n                                                    url = 'https://player.vimeo.com/video/' + match[1];\n\n                                            } else if (data.service == 'dailymotion') {\n                                                var videoId = null;\n                                                var regExp = /^.+dailymotion.com\\/(video|hub)\\/([^_]+)[^#]*(#video=([^_&]+))?/;\n                                                var match = url.match(regExp);\n\n                                                if (match !== null) {\n                                                    if (match[4] !== undefined) {\n                                                        videoId = match[4];\n                                                    }\n                                                    videoId = match[2];\n                                                }\n\n                                                if (videoId)\n                                                    url = 'https://www.dailymotion.com/embed/video/' + videoId;\n\n                                            }\n                                        }\n\n                                        var $embed = $('<div class=\"embed-responsive embed-responsive-16by9\">' +\n                                            '<iframe class=\"embed-responsive-item\" src=\"' + url + '\"></iframe>' +\n                                        '</div>');\n                                        var range = _this._selection.getRangeAt(0);\n                                        range.deleteContents();\n                                        range.insertNode($embed.get(0));\n                                    }\n                                    break;\n\n                                case 'add-image':\n                                    if(_this._selection && value) {\n                                        var $image = $('<img src=\"' + value + '\" />');\n                                        var range = _this._selection.getRangeAt(0);\n                                        range.deleteContents();\n                                        range.insertNode($image.get(0));\n                                    }\n                                    break;\n\n                                case 'insert-html':\n                                    _this._formatDoc('insertHTML', value);\n                                    break;\n\n                                case 'special':\n                                    switch (value) {\n                                        case 'print':\n                                            _this._printDoc();\n                                            break;\n\n                                        case 'clean':\n                                            _this._formatDoc('removeFormat');\n                                            break;\n\n                                        case 'visual':\n                                            if (_this._$content.hasClass('visual')) {\n                                                _this._$toolbar.find('[data-action=\"special\"][data-value=\"visual\"]').removeClass('active');\n                                                _this._$content.removeClass('visual');\n                                                _this._$content.focus();\n                                            } else {\n                                                _this._$toolbar.find('[data-action=\"special\"][data-value=\"visual\"]').addClass('active');\n                                                _this._$content.addClass('visual');\n                                                _this._$content.focus();\n                                            }\n                                            break;\n\n                                        case 'unformat':\n                                            _this._formatDoc('selectAll');\n                                            _this._formatDoc('removeFormat');\n                                            var string = _this._$content.html();\n                                            string = _this._stripTags(string);\n                                            string = string.replace(/(\\r\\n|\\n|\\r)/g, '<!-- br -->');\n                                            string = string.replace(/<!-- br -->/g, '<br/>');\n                                            _this._$content.html(string);\n                                            break;\n                                    }\n                                    break;\n\n                                default:\n                                    if (_this._config.debug)\n                                        console.warn('Unrecognized action: ' + action + ' with value: ' + value);\n\n                                    break;\n                            }\n\n                        }\n\n                    });\n                }\n\n                // On selected content\n                _this._$content.on('mouseup click focus', function (event) {\n                    const $this = $(this);\n\n                    if (_this._popoverIsVisible)\n                        _this._hideAllPopovers();\n\n                    if (event.target.type !== \"text\")\n                        _this._selection = document.getSelection();\n\n                    if (_this._selection.getRangeAt && _this._selection.rangeCount) {\n\n                        if (_this._selection.parentNode) {\n                            var $target = $(_this._selection.parentNode);\n                            _this._updateState($target);\n                        } else if (_this._selection.parentElement) {\n                            var $target = $(_this._selection.parentElement);\n                            _this._updateState($target);\n                        }\n                    }\n\n                    if (_this._config.debug)\n                        console.log('Current selection: ', _this._selection);\n\n                    var $target = $(event.target);\n                    _this._updateState($target);\n                    $this.trigger('change');\n                });\n\n                // On click or keydown from content area\n                _this._$content.on('keydown', function (event) {\n                    const $this = $(this);\n\n                    if(_this._popoverIsVisible)\n                        _this._hideAllPopovers();\n\n                    if(event.target.type !== \"text\")\n                        _this._selection = document.getSelection();\n\n                    if (_this._selection.getRangeAt && _this._selection.rangeCount) {\n\n                        if (_this._selection.parentNode) {\n                            var $target = $(_this._selection.parentNode);\n                            _this._updateState($target);\n                        } else if (_this._selection.parentElement) {\n                            var $target = $(_this._selection.parentElement);\n                            _this._updateState($target);\n                        }\n                    }\n\n                    var $target = $(event.target);\n                    _this._updateState($target);\n                    $this.trigger('change');\n\n                    if (_this._config.debug)\n                        console.log('Keydown fired: ' + event.keyCode);\n\n                });\n\n                // On content change\n                _this._$content.on('change', function(event) {\n                    const $this = $(this);\n                    setTimeout(function() {\n                        if (_this._config.mode == 'editor')\n                            _this._source = $this.html();\n                        else\n                            _this._source = $this.text();\n\n                        if ($(_this._$element).is(\"textarea\"))\n                            _this._$element.html(_this._source);\n                        else\n                            _this._$element.val(_this._source);\n\n                        if (_this._config.debug)\n                            console.log('Content change...');\n\n                    }, 200);\n                });\n\n                // On content lost focus\n                _this._$content.on('blur', function() {\n                    const $this = $(this);\n                    _this._$lastFocus = this;\n                    _this._selection = document.getSelection();\n                    $this.trigger('change');\n\n                    if (_this._config.debug)\n                        console.log('Content lost focus: ', _this._selection);\n\n                });\n\n                // Set focus on content\n                _this._$content.focus();\n\n            }\n\n            _createClass(Editor, {\n                element: {\n                    value: function element() {\n                        var _this = this;\n                        return _this._$element;\n                    }\n                },\n                _replaceAll: {\n                    value: function replaceAll(search, replace, string) {\n                        return string.split(search).join(replace);\n                    }\n                },\n                _stripTags: {\n                    value: function stripTags(string, tags) {\n\n                        var key, allowed_tags = [];\n                        if (tags)\n                            allowed_tags = tags.match(/([a-zA-Z]+)/gi);\n\n                        if (typeof (string) !== 'string')\n                            string = string.toString();\n\n                        var matches = string.match(/(<\\/?[\\S][^>]*>)/gi);\n\n                        for (key in matches) {\n\n                            if (isNaN(key))\n                                continue;\n\n                            var html = matches[key].toString();\n                            var allowed = false;\n\n                            for (key in allowed_tags) {\n\n                                var tag = allowed_tags[key];\n                                var i = html.toLowerCase().indexOf('<'+ tag +'>');\n\n                                if (i != 0)\n                                    i = html.toLowerCase().indexOf('<'+ tag +' ');\n\n                                if (i != 0)\n                                    i = html.toLowerCase().indexOf('</'+ tag );\n\n                                if (i == 0) {\n                                    allowed = true;\n                                    break;\n                                }\n\n                            }\n\n                            if (!allowed)\n                                string = this._replaceAll(html, \"\", string);\n\n                        }\n\n                        return string;\n                    }\n                },\n                _trimSource: {\n                    value: function trimSource(str) {\n                        str = str.replace(/\\s{4,}/g, \"\");\n                        str = str.replace(/\\t/g, ' ');\n                        str = str.toString().trim().replace(/(\\r\\n|\\n|\\r)/g,\"\");\n                        return str;\n                    }\n                },\n                _getPath: {\n                    value: function getPath(node, until, withNodes) {\n\n                        var path, tags = [];\n                        while (node.length) {\n\n                            if (node[0].isEqualNode(until[0]))\n                                break;\n\n                            var realNode = node[0], name = realNode.localName;\n                            var parent = node.parentsUntil(until);\n\n                            if (!name)\n                                break;\n                            else\n                                name = name.toLowerCase();\n\n                            if (withNodes) {\n                                var sameTagSiblings = parent.children(name);\n                                if (sameTagSiblings.length > 1) {\n                                    var allSiblings = parent.children();\n                                    var index = allSiblings.index(realNode) + 1;\n                                    if (index > 1) {\n                                        name += ':nth-child(' + index + ')';\n                                    }\n                                }\n                            }\n\n                            tags.push(name);\n\n                            var id = $(realNode).attr(\"id\");\n                            if (id)\n                                name += \"#\" + id;\n\n                            var classname = $(realNode).attr(\"class\");\n                            if (classname)\n                                name += \".\" + classname.replace(/\\./g, '.');\n\n                            path = name + (path ? ' > ' + path : '');\n                            node = parent;\n                        }\n\n                        return {\n                            path: path,\n                            tags: tags\n                        };\n                    }\n                },\n                _getTextStat: {\n                    value: function getTextStat(el) {\n\n                        var words = 0, length = 0, chars = 0, normalizedValue;\n                        var isContentEditable = el && el.contentEditable;\n\n                        if (isContentEditable)\n                            normalizedValue = el.innerText.replace(/\\r\\n/g, \"\\n\");\n                        else\n                            normalizedValue = el.value.replace(/\\r\\n/g, \"\\n\");\n\n                        words = this._stripTags(normalizedValue).split(' ').length;\n                        length = normalizedValue.length;\n                        chars = this._trimSource(normalizedValue.replace(/\\s/g, \"\")).length;\n\n                        return {\n                            words: words,\n                            length: length,\n                            chars: chars\n                        }\n                    }\n                },\n                _getTextPosition: {\n                    value: function getCursorPosition(el) {\n\n                        var line = 0, start = 0, end = 0, selected = 0, normalizedValue, range, textInputRange, len, endRange;\n                        var isContentEditable = el && el.contentEditable;\n\n                        if (\"selectionStart\" in el && document.activeElement == el) {\n\n                            start = el.selectionStart;\n                            end = el.selectionEnd;\n                            normalizedValue = el.value.replace(/\\r\\n/g, \"\\n\");\n                            line = normalizedValue.substr(0, el.selectionStart).split(\"\\n\").length;\n\n                        } else if (isContentEditable) {\n\n                            start = window.getSelection().getRangeAt(0).startOffset;\n                            end = window.getSelection().getRangeAt(0).endOffset;\n\n                            normalizedValue = el.innerText.replace(/\\r\\n/g, \"\\n\");\n                            line = (normalizedValue.substr(0, el.selectionStart).split(\"\\n\").length - 1);\n\n                            if(line == 0)\n                                line = 1;\n\n                        } else {\n\n                            range = this._selection.createRange();\n\n                            if (range && range.parentElement() == el) {\n                                len = el.value.length;\n                                normalizedValue = el.value.replace(/\\r\\n/g, \"\\n\");\n\n                                // Create a working TextRange that lives only in the input\n                                textInputRange = el.createTextRange();\n                                textInputRange.moveToBookmark(range.getBookmark());\n\n                                // Check if the start and end of the selection are at the very end\n                                // of the input, since moveStart/moveEnd doesn't return what we want\n                                // in those cases\n                                endRange = el.createTextRange();\n                                endRange.collapse(false);\n\n                                if (textInputRange.compareEndPoints(\"StartToEnd\", endRange) > -1) {\n                                    start = end = len;\n                                } else {\n                                    start = -textInputRange.moveStart(\"character\", -len);\n                                    start += normalizedValue.slice(0, start).split(\"\\n\").length - 1;\n\n                                    if (textInputRange.compareEndPoints(\"EndToEnd\", endRange) > -1) {\n                                        end = len;\n                                    } else {\n                                        end = -textInputRange.moveEnd(\"character\", -len);\n                                        end += normalizedValue.slice(0, end).split(\"\\n\").length - 1;\n                                    }\n                                }\n                            }\n                        }\n\n                        selected = (this._selection.toString()).length;\n\n                        return {\n                            line: line,\n                            start: start,\n                            end: end,\n                            selected: selected\n                        }\n                    }\n                },\n                _formatDoc: {\n                    value: function formatDoc(command, value) {\n                        document.execCommand(command, false, value);\n                        this._$content.focus();\n                    }\n                },\n                _printDoc: {\n                    value: function printDoc() {\n                        var print = window.open(\"\",\"_blank\",\"width=450,height=470,left=400,top=100,menubar=yes,toolbar=no,location=no,scrollbars=yes\");\n                        print.document.open();\n                        print.document.write(\"<!doctype html><html><head><title>Print<\\/title><\\/head><body onload=\\\"print();\\\">\" + this._$content.get(0).innerHTML + \"<\\/body><\\/html>\");\n                        print.document.close();\n                    }\n                },\n                _hideAllPopovers: {\n                    value: function hideAllPopovers() {\n                        this._$toolbar.find('.popover').each(function() {\n                            $(this).popover('hide');\n                        });\n                        this._popoverIsVisible = false;\n                    }\n                },\n                _detectLanguage: {\n                    value: function detectLanguage() {\n\n                        var language = null;\n                        if (navigator.languages && navigator.languages.length) {\n                            language = navigator.languages[0];\n                        } else {\n                            language = navigator.userLanguage || navigator.language || navigator.browserLanguage || 'en';\n                        }\n\n                        return language.toLowerCase();\n\n                    }\n                },\n                _translate: {\n                    value: function translate(string) {\n\n                        var _this = this;\n                        var language = _this._config.language;\n\n                        if (typeof (language) === \"undefined\")\n                            language = _this._detectLanguage();\n\n                        if (_this._config.translations.hasOwnProperty(language)) {\n                            if (_this._config.translations[language][string.toString()]) {\n                                string = _this._config.translations[language][string.toString()];\n                            }\n                        }\n\n                        return string.toLocaleString();\n                    }\n                },\n                _buildTollbarButton: {\n                    value: function buildTollbarButton(action, value, icon, hotkey, tooltip, content) {\n\n                        var _this = this;\n                        var selection = _this._selection;\n                        var $button = $('<button type=\"button\" class=\"btn btn-default\" tabindex=\"-1\" />');\n\n                        if (action)\n                            $button.attr('data-action', action);\n\n                        if (value)\n                            $button.attr('data-value', value);\n\n                        if (hotkey)\n                            $button.attr('data-hotkey', hotkey);\n\n                        if (tooltip) {\n                            $button.tooltip({\n                                html: true,\n                                placement: 'top',\n                                container: 'body',\n                                title: _this._translate(tooltip.toString().trim())\n                            });\n                        }\n\n                        if (content) {\n                            $button.popover({\n                                html: true,\n                                trigger: 'manual',\n                                viewport: 'body',\n                                placement: 'bottom',\n                                content: function() {\n\n                                    if (typeof (content) === \"object\")\n                                        return content;\n                                    else\n                                        return $(content);\n\n                                }\n                            }).on('shown.bs.popover', function(event) {\n\n                                var popoverId = $(event.target).attr('aria-describedby');\n                                var $popover = _this._$toolbar.find('#'+popoverId);\n\n\n                                var selection = _this._selection;\n                                var range = selection.getRangeAt(0);\n                                if(selection && range) {\n                                    $popover.find('input').on('blur', function(event) {\n                                        if (event.target.type == \"text\") {\n                                            if (_this._$lastFocus) {\n                                                setTimeout(function() {\n                                                    _this._$lastFocus.focus();\n                                                    _this._selectText(selection, range);\n                                                }, 50);\n                                            }\n                                        }\n                                        return false;\n                                    });\n                                }\n\n                                $popover.on('click', function(event) {\n                                    event.preventDefault();\n\n                                    if (_this._config.debug) {\n                                        console.log('Popover event target type: ' + event.target.type);\n                                        console.log('Popover event target tag: ' + event.target.tagName);\n                                    }\n\n                                    if (!(event.target.type) && !(event.target.tagName.toLowerCase() == 'a'))\n                                        return;\n\n                                    if ($(event.target).get(0).hasAttribute('data-action')) {\n                                        $popover.popover('hide');\n                                    }\n\n                                });\n\n                                if ($popover.find('.table-grid').length) {\n                                    $popover.find('.table-grid tr > td').hover(function() {\n                                        $(this).addClass('selected');\n                                        $(this).prevAll().addClass('selected');\n                                        $(this).parent().prevAll().find('td:lt('+ ($(this).index() + 1) + ')').addClass('selected');\n                                    }, function() {\n                                        $(this).removeClass('selected');\n                                        $(this).prevAll().removeClass('selected');\n                                        $(this).parent().prevAll().find('td:lt('+ ($(this).index() + 1) + ')').removeClass('selected');\n                                    });\n                                }\n\n                            }).on('click', function(event) {\n\n                                if (_this._config.debug) {\n                                    console.log('Element event target type: ' + event.target.type);\n                                    console.log('Element event target tag: ' + event.target.tagName);\n                                }\n\n                                event.preventDefault();\n                                event.stopPropagation();\n\n                                if(_this._popoverIsVisible)\n                                    _this._hideAllPopovers();\n\n                                $button.popover('show');\n                                _this._popoverIsVisible = true;\n                            });\n\n                        } else {\n                            $button.on('click', function(event) {\n                                event.preventDefault();\n                                _this._hideAllPopovers();\n                            });\n                        }\n\n                        if (icon)\n                            $button.append('<span class=\"' + icon + '\" />');\n\n                        return $button;\n                    }\n                },\n                _buildTollbarDropdown: {\n                    value: function buildTollbarDropdown(action, list, label, tooltip) {\n\n                        var $dropdown = $('<div class=\"dropdown\" />');\n                        var $dropdownBtn = $('<button type=\"button\" class=\"btn btn-default dropdown-toggle\" data-toggle=\"dropdown\" />');\n                        var $dropdownMenu = $('<ul class=\"dropdown-menu\" />');\n                        var $dropdownItem = $('<li />');\n                        var $dropdownLink = $('<a href=\"#\" tabindex=\"-1\" />');\n\n                        if (typeof (list) == \"object\") {\n\n                            $.each(list, function(index, elem) {\n\n                                var $link = $dropdownLink.clone();\n                                var $item = $dropdownItem.clone();\n\n                                if (typeof (elem) == 'object') {\n\n                                    if (elem['action'])\n                                        $link.attr('data-action', elem['action']);\n\n                                    if (elem['value'])\n                                        $link.attr('data-value', elem['value']);\n\n                                    if (elem['wrap'])\n                                        $link.html($(elem['wrap']).text(index));\n                                    else\n                                        $link.text(index);\n\n                                    if (elem['style'])\n                                        $link.attr('style', elem['style'].toString());\n\n                                    if(index == label)\n                                        $item.addClass('active');\n\n                                    $item.append($link);\n                                    $dropdownMenu.append($item);\n\n                                } else {\n\n                                    $link.text(elem);\n                                    $link.attr('data-action', action);\n                                    $link.attr('data-value', elem);\n\n                                    if(elem == label)\n                                        $item.addClass('active');\n\n                                    $item.append($link);\n                                    $dropdownMenu.append($item);\n                                }\n\n                                $link.on('click', function (e) {\n                                    e.preventDefault();\n                                });\n\n                            });\n                        }\n\n                        if (label)\n                            $dropdownBtn.text(label + ' ');\n                        else\n                            $dropdownBtn.text('Dropdown ');\n\n                        $dropdownBtn.append('<b class=\"caret\" />');\n\n                        if (tooltip) {\n                            $dropdownBtn.tooltip({\n                                html: true,\n                                placement: 'top',\n                                title: this._translate(tooltip.toString().trim())\n                            });\n                        }\n\n                        $dropdown.append($dropdownBtn);\n                        $dropdown.append($dropdownMenu);\n                        return $dropdown;\n                    }\n                },\n                _buildColorPalette: {\n                    value: function buildColorPalette(palette, action, reset) {\n                        var content = '';\n                        $.each(palette, function (outer, colors) {\n                            content += '<table class=\"color-palette\"><tr>';\n                            $.each(colors, function (inner, color) {\n                                content += '<td><a href=\"#\" data-action=\"' + action + '\" data-value=\"' + color + '\" style=\"background-color: ' + color + '\">&nbsp;</a></td>'\n                                content += ((parseInt(inner) + 1)%10 ? '' : '</tr>');\n                                content += ((parseInt(inner) + 1)%10 ? '' : '<tr>');\n                            });\n                            content += '</tr></table>';\n                        });\n\n                        if(reset)\n                            content += '<p><a href=\"#\" class=\"btn btn-sm btn-block\" data-action=\"' + action + '\" data-value=\"unset\">Reset color</a></p>';\n\n                        return content;\n                    }\n                },\n                _buildTableGrid: {\n                    value: function buildColorPalette() {\n\n                        var content = '<table class=\"table-grid\">';\n\n                        for(var row = 1; row <= 6; row++) {\n\n                            content += '<tr>';\n\n                            for(var column = 1; column <= 8; column++) {\n                                content += '<td><a href=\"#\" data-action=\"insert-table\" data-value=\"' + row + '|'+ column +'\">&nbsp;</a></td>'\n                            }\n\n                            content += '</tr>';\n                        }\n\n                        content += '</table>';\n                        return content;\n                    }\n                },\n                _buildLetterSpacingList: {\n                    value: function buildLetterSpacingList() {\n                        var content = '<ul class=\"nav nav-pills nav-stacked\">\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"-5\">-5</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"-3\">-3</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"-2\">-2</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"-1\">-1</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"0\">0</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"1\">1</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"2\">2</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"3\">3</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"5\">5</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"8\">8</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"10\">10</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"12\">12</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"15\">15</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"25\">25</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"letter-spacing\" data-value=\"50\">50</a></li>\\n' +\n                            '</ul>';\n\n                        return content;\n                    }\n                },\n                _buildLineHeightList: {\n                    value: function buildLineHeightList() {\n                        var content = '<ul class=\"nav nav-pills nav-stacked\">\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"line-height\" data-value=\"0.5\">0.5</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"line-height\" data-value=\"1.0\">1.0</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"line-height\" data-value=\"1.15\">1.15</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"line-height\" data-value=\"1.5\">1.5</a></li>\\n' +\n                            '  <li role=\"presentation\"><a href=\"#\" data-action=\"line-height\" data-value=\"2.0\">2.0</a></li>\\n' +\n                            '</ul>';\n\n                        return content;\n                    }\n                },\n                _selectText: {\n                    value: function selectText(selection, range) {\n\n                        if(!selection)\n                            selection = document.getSelection();\n\n                        if(!range)\n                            range = selection.getRangeAt(0);\n\n                        selection.removeAllRanges();\n                        selection.addRange(range);\n                    }\n                },\n                _buildEmojiList: {\n                    value: function buildEmojiList() {\n                        var emojis = this._config.emojiDefault;\n                        if(emojis.length > 0) {\n                            var maxRows = Math.round(emojis.length / 8)+1;\n                            var content = '<table class=\"emojis-list\">';\n                            for(var row = 1, index = 0; row <= maxRows; row++) {\n                                content += '<tr>';\n\n                                for(var column = 1; column <= 8; column++) {\n\n                                    if(index == emojis.length)\n                                        break;\n\n                                    content += '<td><a href=\"#\" data-action=\"insert-html\" data-value=\"' + emojis[index].toString() + '\">'+ emojis[index] +'</a></td>';\n                                    index++;\n                                }\n                                content += '</tr>';\n                            }\n                            content += '</table>';\n                            return content;\n\n                        } else {\n                            return false;\n                        }\n                    }\n                },\n                _buildSymbolsList: {\n                    value: function buildSymbolsList() {\n\n                        var symbols = this._config.symbolsDefault;\n\n                        if(symbols.length > 0) {\n                            var maxRows = Math.round(symbols.length / 10);\n                            var content = '<table class=\"symbols-list\">';\n                            for(var row = 1, index = 0; row <= maxRows; row++) {\n                                content += '<tr>';\n\n                                for(var column = 1; column <= 10; column++) {\n\n                                    if(index == symbols.length)\n                                        break;\n\n                                    content += '<td><a href=\"#\" data-action=\"insert-html\" data-value=\"' + symbols[index] + '\" style=\"min-width:16px;text-align:center;\">'+ symbols[index] +'</a></td>';\n                                    index++;\n                                }\n                                content += '</tr>';\n                            }\n                            content += '</table>';\n\n                            return content;\n\n                        } else {\n                            return false;\n                        }\n\n                    }\n                },\n                _buildDdropdown: {\n                    value: function buildDdropdown(dropdownId, buttonText, buttonCaret, menuItems, defaultValue, dataAttr) {\n\n                        if (buttonText == null)\n                            buttonText = 'Not set';\n\n                        if (buttonCaret == null)\n                            buttonCaret = '<span class=\"caret\"></span>';\n\n                        if (menuItems == null)\n                            buttonCaret = {};\n\n                        if (dataAttr == null)\n                            dataAttr = 'data-value';\n\n                        var $dropdown = $('<div class=\"dropdown\" />');\n                        var $dropdownButton = $('<button type=\"button\" class=\"btn btn-block btn-default dropdown-toggle\" data-toggle=\"dropdown\" aria-haspopup=\"true\" aria-expanded=\"false\" />');\n                        $dropdownButton.attr('id', dropdownId);\n                        $dropdownButton.html(buttonText + ' ' + buttonCaret);\n\n                        // Build dropdown menu\n                        var $dropdownMenu = $('<ul class=\"dropdown-menu\" aria-labelledby=\"' + dropdownId + '\" />');\n\n                        if (!defaultValue)\n                            $dropdownMenu.append('<li class=\"active\"><a href=\"#\">' + buttonText + '</a></li>');\n\n                        for (let [id, name] of Object.entries(menuItems)) {\n                            if (id.toString() == 'separator') {\n                                $dropdownMenu.append('<li role=\"separator\" class=\"divider\"></li>');\n                            } else if (defaultValue == id.toString()) {\n                                $dropdownMenu.append('<li class=\"active\"><a href=\"#\" ' + dataAttr + '=\"' + id + '\">' + name + '</a></li>');\n                            } else {\n                                $dropdownMenu.append('<li><a href=\"#\" ' + dataAttr + '=\"' + id + '\">' + name + '</a></li>');\n                            }\n                        }\n\n                        // Click by dropdown menu items\n                        $dropdownMenu.find('li').on('click', function(event) {\n                            $dropdownMenu.find('li').removeClass('active');\n                            $(this).addClass('active');\n                            $dropdownButton.html($(this).text() + ' ' + buttonCaret);\n                        });\n\n                        $dropdown.append($dropdownButton);\n                        $dropdown.append($dropdownMenu);\n                        return $dropdown;\n                    }\n                },\n                _buildUrlForm: {\n                    value: function buildUrlForm(type) {\n\n                        var _this = this;\n\n                        if (this._config.debug)\n                            console.log('Build URL form for type: ' + type);\n\n\n                        var $form = $('<form class=\"form-horizontal\" />');\n                        var $formGroup = $('<div class=\"form-group\" />');\n                        var $container = $('<div class=\"col-xs-12 col-sm-12\" />');\n                        var $inputGroup = $('<div class=\"input-group input-group-sm\" />');\n                        var $dropdown = $('<div class=\"dropdown\" />');\n                        var $dropdownButton = $('<button type=\"button\" class=\"btn btn-block btn-secondary dropdown-toggle\" data-toggle=\"dropdown\" aria-haspopup=\"true\" aria-expanded=\"false\" />');\n\n                        if (type == \"image\") {\n                            $inputGroup.append('<span class=\"input-group-addon\">Image:</span>');\n                        } else if (type == \"video\") {\n                            var $dropdown = _this._buildDdropdown('videoServices', 'YouTube', null, videoServices, 'youtube', 'data-service');\n                            $dropdown.attr('class', 'input-group-btn');\n                            $dropdown.find('.btn[data-toggle=\"dropdown\"]').toggleClass('btn-default', 'btn-secondary');\n                            $inputGroup.append($dropdown);\n                        } else if (type == \"link\") {\n                            var $dropdown = _this._buildDdropdown('urlSchemes', 'https://', null, urlSchemes, 'https', 'data-scheme');\n                            $dropdown.attr('class', 'input-group-btn');\n                            $dropdown.find('.btn[data-toggle=\"dropdown\"]').toggleClass('btn-default', 'btn-secondary');\n                            $inputGroup.append($dropdown);\n                        }\n\n                        var $input = $('<input type=\"text\" class=\"form-control\" placeholder=\"Type your URL...\" />');\n                        if (type == \"image\") {\n                            $input.attr('id', \"imageUrl\");\n                        } else if (type == \"video\") {\n                            $input.attr('id', \"videoUrl\");\n                        } else if (type == \"link\") {\n                            $input.attr('id', \"urlInput\");\n                        }\n                        $inputGroup.append($input);\n\n                        var $buttonWrap = $('<span class=\"input-group-btn\" />');\n                        var $button = $('<button type=\"button\" class=\"btn btn-block btn-primary\">Add</button>');\n\n                        if (type == \"image\") {\n                            $button.attr('data-action', \"add-image\");\n                        } else if (type == \"video\") {\n                            $button.attr('data-action', \"add-video\");\n                        } else {\n                            $button.attr('data-action', \"add-url\");\n                        }\n\n                        $button.on('click', function (event) {\n\n                            var action = $(event.target).data('action');\n                            $(event.target).data('value', $input.val());\n\n                            if (action == \"add-url\") {\n                                var urlScheme = $form.find('[aria-labelledby=\"urlSchemes\"] li.active a[data-scheme]').first().data('scheme');\n                                $(event.target).data('scheme', (urlScheme) ? urlScheme : null);\n\n                                var urlTile = $form.find('#urlTile').val();\n                                $(event.target).data('title', (urlTile) ? urlTile : null);\n\n                                var urlClass = $form.find('#urlClass').val();\n                                $(event.target).data('class', (urlClass) ? urlClass : null);\n\n                                var urlLinkTarget = $form.find('[aria-labelledby=\"urlLinkTarget\"] li.active a[data-target]').first().data('target');\n                                $(event.target).data('target', (urlLinkTarget) ? urlLinkTarget : null);\n\n                                var urlLinkRel = $form.find('[aria-labelledby=\"urlLinkRel\"] li.active a[data-rel]').first().data('rel');\n                                $(event.target).data('relation', (urlLinkRel) ? urlLinkRel : null);\n\n                            } else if (action == \"add-video\") {\n                                var videoService = $form.find('[aria-labelledby=\"videoServices\"] li.active a[data-service]').first().data('service');\n                                $(event.target).data('service', (videoService) ? videoService : null);\n                            }\n\n                            console.log($button.data());\n                        });\n\n                        $buttonWrap.append($button);\n                        $inputGroup.append($buttonWrap);\n                        $container.append($inputGroup);\n                        $formGroup.append($container);\n                        $form.append($formGroup);\n\n                        if (type == \"link\") {\n                            var content = '';\n                            content += '<div class=\"form-group form-group-sm\">';\n                            content += '    <div class=\"form-row\">';\n                            content += '        <label class=\"col-xs-12 col-sm-4\">Tile:</label>';\n                            content += '        <div class=\"col-xs-12 col-sm-8\">';\n                            content += '            <input type=\"text\" class=\"form-control\" id=\"urlTile\" placeholder=\"Title of link...\" />';\n                            content += '        </div>';\n                            content += '    </div>';\n                            content += '</div>';\n\n                            content += '<div class=\"form-group form-group-sm\">';\n                            content += '    <div class=\"form-row\">';\n                            content += '        <label class=\"col-xs-12 col-sm-4\">Class:</label>';\n                            content += '        <div class=\"col-xs-12 col-sm-8\">';\n                            content += '            <input type=\"text\" class=\"form-control\" id=\"urlClass\" placeholder=\"CSS class name...\" />';\n                            content += '        </div>';\n                            content += '    </div>';\n                            content += '</div>';\n\n                            $form.append(content);\n\n                            // Link target\n                            var $formGroup = $('<div class=\"form-group form-group-sm\" />');\n                            var $inputGroup = $('<div class=\"form-row input-group-sm\" />');\n                            var $inputLabel = $('<label class=\"col-xs-12 col-sm-4\" />');\n                            $inputLabel.text('Target:');\n                            $inputGroup.append($inputLabel);\n\n                            var $dropdown = _this._buildDdropdown('urlLinkTarget', 'Not set', null, urlLinkTarget, null, 'data-target');\n                            $dropdown.addClass('col-xs-12 col-sm-8');\n                            $dropdown.find('.btn[data-toggle=\"dropdown\"]').addClass('btn-sm');\n\n                            $inputGroup.append($dropdown);\n                            $formGroup.append($inputGroup);\n                            $form.append($formGroup);\n\n                            // Link rel\n                            var $formGroup = $('<div class=\"form-group form-group-sm\" />');\n                            var $inputGroup = $('<div class=\"form-row input-group-sm\" />');\n                            var $inputLabel = $('<label class=\"col-xs-12 col-sm-4\" />');\n                            $inputLabel.text('Relation:');\n                            $inputGroup.append($inputLabel);\n\n                            var $dropdown = _this._buildDdropdown('urlLinkRel', 'Not set', null, urlLinkRel, null, 'data-rel')\n                            $dropdown.addClass('col-xs-12 col-sm-8');\n                            $dropdown.find('.btn[data-toggle=\"dropdown\"]').addClass('btn-sm');\n\n                            $inputGroup.append($dropdown);\n                            $formGroup.append($inputGroup);\n                            $form.append($formGroup);\n\n                        }\n\n                        return $form;\n                    }\n                },\n                _generateTable: {\n                    value: function generateTable(rows, columns) {\n\n                        rows = parseInt(rows) + 1;\n                        columns = parseInt(rows);\n\n                        if(!columns) columns = 1;\n\n                        var content = '<table class=\"table\">';\n\n                        for(var row = 1; row <= rows; row++) {\n\n                            if (row == 1)\n                                content += '<thead>';\n                            else if (row == ((rows - row) - 1))\n                                content += '<tbody>';\n\n                            content += '<tr>';\n\n                            for(var column = 1; column <= columns; column++) {\n\n                                if (row == 1)\n                                    content += '<th>Header ' + column + '</th>';\n                                else\n                                    content += '<td>&nbsp;</td>';\n                            }\n\n                            content += '</tr>';\n\n                            if (row == 1)\n                                content += '</thead>';\n                            else if (row == rows)\n                                content += '</tbody>';\n                        }\n\n                        content += '</table>';\n                        return content;\n                    }\n                },\n                _updateState: {\n                    value: function updateState($target, reset) {\n                        var _this = this;\n                        if (_this._config.mode == 'editor') {\n                            var statInfo = _this._getTextStat(_this._$content.get(0));\n                            var pathInfo = _this._getPath($target, _this._$content, false);\n\n                            if(!reset) {\n                                switch (pathInfo['tags'][0]) {\n                                    case 'b' :\n                                        _this._$toolbar.find('[data-action=\"text\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"text\"][data-value=\"bold\"]').addClass('active');\n                                        break;\n\n                                    case 'u' :\n                                        _this._$toolbar.find('[data-action=\"text\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"text\"][data-value=\"underline\"]').addClass('active');\n                                        break;\n\n                                    case 'sub' :\n                                        _this._$toolbar.find('[data-action=\"text\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"text\"][data-value=\"subscript\"]').addClass('active');\n                                        break;\n\n                                    case 'sup' :\n                                        _this._$toolbar.find('[data-action=\"text\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"text\"][data-value=\"superscript\"]').addClass('active');\n                                        break;\n\n                                    case 'i' :\n                                        _this._$toolbar.find('[data-action=\"text\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"text\"][data-value=\"italic\"]').addClass('active');\n                                        break;\n\n                                    case 'a' :\n                                        _this._$toolbar.find('[data-action=\"insert\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"insert\"][data-value=\"link\"]').addClass('active');\n                                        break;\n\n                                    case 'p' :\n                                        _this._$toolbar.find('[data-action=\"align\"]').removeClass('active');\n\n                                        if ($target.css('text-align') == 'center')\n                                            _this._$toolbar.find('[data-action=\"align\"][data-value=\"center\"]').addClass('active');\n                                        else if ($target.css('text-align') == 'right')\n                                            _this._$toolbar.find('[data-action=\"align\"][data-value=\"right\"]').addClass('active');\n                                        else if ($target.css('text-align') == 'justify')\n                                            _this._$toolbar.find('[data-action=\"align\"][data-value=\"justify\"]').addClass('active');\n                                        else\n                                            _this._$toolbar.find('[data-action=\"align\"][data-value=\"left\"]').addClass('active');\n\n                                        break;\n\n                                    default :\n                                        _this._$toolbar.find('[data-action=\"text\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"align\"]').removeClass('active');\n                                        _this._$toolbar.find('[data-action=\"insert\"]').removeClass('active');\n                                        break;\n                                }\n                            } else {\n                                _this._$toolbar.find('button[data-action]').removeClass('active');\n                            }\n\n                            _this._$statusbar.path.text(pathInfo['path']);\n                            _this._$statusbar.stat.text('Length: ' + statInfo['length'] + ', chars: ' + statInfo['chars'] + ', words: ' +  statInfo['words']);\n\n                        } else {\n\n                            var position = _this._getTextPosition(_this._$content.get(0));\n                            _this._$statusbar.path.empty();\n\n                            if(parseInt(position['selected']) > 0)\n                                _this._$statusbar.stat.text('Line: ' + position['line'] + ', column: ' + position['end'] + ', selected: ' + position['selected']);\n                            else\n                                _this._$statusbar.stat.text('Line: ' + position['line'] + ', column: ' + position['end']);\n\n                        }\n\n                        return _this._$element;\n\n                    }\n                },\n            }, {\n                Default: {\n                    get: function() {\n                        return defaults;\n                    }\n                },\n                _jQueryInterface: {\n                    value: function _jQueryInterface(config) {\n                        let _this = this;\n                        config = config || {};\n\n                        if (/destroy|hide/.test(config)) {\n                            let element = $(_this);\n                            element.removeClass('hide');\n\n                            let editor = element.parent('.wysiwyg-editor');\n                            editor.replaceWith(element);\n\n                            return;\n                        }\n\n                        return _this.each(function() {\n                            let $this = $(_this);\n                            let _config = $.extend({}, WYSIWYG.Default, $this.data(), typeof config === \"object\" && config);\n                            new Editor(_this, _config);\n                        });\n                    }\n                }\n            });\n\n            return Editor;\n\n        })();\n\n        $.fn[className] = Editor._jQueryInterface;\n        $.fn[className].Constructor = Editor;\n        $.fn[className].noConflict = function() {\n            $.fn[className] = _jQueryNoConflict;\n            return Editor._jQueryInterface;\n        };\n\n        return Editor;\n\n    })(jQuery);\n}(jQuery);"]}